使用计算属性cale_乾坤大挪移!React 也能“用上”computed 属性

本文为第 10 篇原创,扫码关注公众号「政采云前端团队」,获取更多不掺水的原创好文~

使用计算属性cale_乾坤大挪移!React 也能“用上”computed 属性_第1张图片
使用计算属性cale_乾坤大挪移!React 也能“用上”computed 属性_第2张图片

前言,关于计算属性

初次见到计算属性一词,是在 Vue 官方文档 《计算属性和侦听器》 一节中,文章中是这样描述计算属性的:

模板内的表达式非常便利,但是设计它们的初衷是用于简单运算的。在模板中放入太多的逻辑会让模板过重且难以维护。

回想我们编写的 React 代码,是否也在 JSX(render 函数)中放入了太多的逻辑导致 render 函数过于庞大,难以维护?

React 中的计算属性

说到 React 之前,我们先看下 Vue,在 Vue 中,计算属性主要有以下两点特性:

  1. 计算属性以声明的方式创建依赖关系,依赖的 data 或 props 变更会触发重新计算并自动更新。
  2. 计算属性是基于它们的响应式依赖进行缓存的。

而在 React 中,计算属性也是经常可见,相信各位熟悉 React 的读者都写过类似下面的代码:

import React, { Fragment, Component } from 'react';class Example extends Component {  state = {    firstName: '',    lastName: '',  };  render() {    // 在 render 函数中处理逻辑    const { firstName, lastName } = this.state;    const fullName = `${firstName} ${lastName}`;    return {fullName};  }}

在上面的代码里,render 函数里的 fullName 依赖了 props 中的 firstName 和 lastName 。firstName 或 lastName 变更之后,变量 fullName 都会自动更新。其实现原理是 props 以及 state 的变化会导致 render 函数调用,进而重新计算衍生值。

虽然能实现计算,但我们还是把计算逻辑放入了 render 函数导致了它的臃肿,这并不优雅。更好的做法是把计算逻辑抽出来,简化 render 函数逻辑:

class Example extends Component {  state = {    firstName: '',    lastName: '',  };  // 把 render 中的逻辑抽成函数,减少render函数的臃肿  renderFullName() {    const { firstName, lastName } = this.state;    return `${firstName} ${lastName}`;  }  render() {    const fullName = this.renderFullName();    return {fullName};  }}

如果你对 Vue 很了解,你肯定知道其 computed 计算属性,底层是使用了getter,只不过是对象的 getter。那么在 React 中,我们也可以使用类的 getter 来实现计算属性:

class Example extends Component {  state = {    firstName: '',    lastName: '',  };  // 通过getter而不是函数形式,减少变量  get fullName() {    const { firstName, lastName } = this.state;    return `${firstName} ${lastName}`;  }  render() {    return {this.fullName};  }}

进一步,使用 memoization 优化计算属性

上文有提到在 Vue 中计算属性对比函数执行,会有缓存,减少计算。因为计算属性只有在它的相关依赖发生改变时才会重新求值。

这就意味着只要 firstName 和 lastName 还没有发生改变,多次访问 fullName 计算属性会立即返回之前的计算结果,而不必再次执行函数。

对比之下,React 的 getter 是否也有缓存这个优势??? 答案是:没有。React 中的 getter 并没有做缓存优化

不过不用失望,我们可以使用记忆化技术(memoization)来优化我们的计算属性,达到和 Vue 中计算属性一样的效果。我们需要在项目中引入 memoize-one 库,代码如下:

import memoize from 'memoize-one';import React, { Fragment, Component } from 'react';class Example extends Component {  state = {    firstName: '',    lastName: '',  };  // 如果和上次参数一样,`memoize-one` 会重复使用上一次的值。  getFullName = memoize((firstName, lastName) => `${firstName} ${lastName}`);  get fullName() {    return this.getFullName(this.state.firstName, this.state.lastName);  }  render() {    return {this.fullName};  }}

再进一步,使用 React Hooks 优化计算属性

上文在 React 中使用了 memoize-one 库实现了类似 Vue 计算属性(computed)的效果 —— 基于依赖缓存计算结果。得益于React 16.8 新推出的 Hooks 特性,我们可以对逻辑进行更优雅的封装,对 Hooks 还不够了解的小伙伴可以先阅读我们团队另一篇文章 《看完这篇,你也能把 React Hooks 玩出花》

此处,我们需要用到 useMemo。官方对 useMemo 的介绍在 这里,详情请移步查看。简单的说,就是我们传入一个 回调函数 和一个 依赖列表,React 会在依赖列表中的值变化时,调用这个回调函数,并将回调函数返回的结果进行缓存:

import React, { useState, useMemo } from 'react';function Example(props) {  const [firstName, setFirstName] = useState('');  const [lastName, setLastName] = useState('');  // 使用 useMemo 函数缓存计算过程  const renderFullName = useMemo(() => `${firstName} ${lastName}`, [    firstName,    lastName,  ]);  return 
{renderFullName}
;}

总结

本文介绍了在 React 中如何实现类似 Vue 计算属性(computed)的效果 —— 基于依赖缓存计算结果,实现逻辑计算与视图渲染的解耦,降低 render 函数的复杂度。

从业务开发角度来讲,Vue 提供的 API 极大地提高了开发效率。React 虽然在某些场景下,没有官方的同类原生 API 支持,但得益于活跃的社区,工作中遇到的问题总能找到解决方案。且在摸索这些解决方案的同时,我们还能学习到诸多经典的编程思想,帮助我们更合理的运用框架,用技术解决业务问题。

招贤纳士

政采云前端团队(ZooTeam),一个年轻富有激情和创造力的前端团队,隶属于政采云产品研发部,Base 在风景如画的杭州。团队现有 50 余个前端小伙伴,平均年龄 27 岁,近 3 成是全栈工程师,妥妥的青年风暴团。成员构成既有来自于阿里、网易的“老”兵,也有浙大、中科大、杭电等校的应届新人。团队在日常的业务对接之外,还在物料体系、工程平台、搭建平台、性能体验、云端应用、数据分析及可视化等方向进行技术探索和实战,推动并落地了一系列的内部技术产品,持续探索前端技术体系的新边界。

如果你想改变一直被事折腾,希望开始能折腾事;如果你想改变一直被告诫需要多些想法,却无从破局;如果你想改变你有能力去做成那个结果,却不需要你;如果你想改变你想做成的事需要一个团队去支撑,但没你带人的位置;如果你想改变既定的节奏,将会是“5 年工作时间 3 年工作经验”;如果你想改变本来悟性不错,但总是有那一层窗户纸的模糊… 如果你相信相信的力量,相信平凡人能成就非凡事,相信能遇到更好的自己。如果你希望参与到随着业务腾飞的过程,亲手推动一个有着深入的业务理解、完善的技术体系、技术创造价值、影响力外溢的前端团队的成长历程,我觉得我们该聊聊。任何时间,等着你写点什么,发给 [email protected]

使用计算属性cale_乾坤大挪移!React 也能“用上”computed 属性_第3张图片

你可能感兴趣的:(使用计算属性cale)