项目性能优化 React

最近项目迭代差不多,老板过来让我优化优化项目,什么优化,老板不会是想把我给优化掉吗,这不是难到我了嘛。但是也无奈只能硬着头皮搞,一把梭。

函数缓存

使用react项目嘛优化方案肯定也听的蛮多什么useCallback、useMemo、memo啊这些常规优化手段,但是使用这些优化手段之后发现页面还是很不流畅,复杂点的页面还是会重复渲染多次。

js 复制代码
const Son = React.memo(({ onChange }) => {
 
 return (
   <div onClick={onChange}>
    我爱加班、我爱加班!
   </div>
 )
})

const Parent = () => {
  const [value, setValue] = useState(0);
  
  const hanldeChange = useCallback(() => {
     const newValue = value + 5;
     setValue(newValue);
  }, [value]);
  
  // 一些复杂计算可以放入 这里只是案例,简单计算不用包裹useMemo ,并且useMemo缓存的是针对子组件的。
  const count = useMemo(() => {
    return value * 10000;
  }, [value]);
  
  return (
    <div>
      <p>喜提{ count }个bug</p>
      <button onClick={() => setValue(value++)}>我爱加班</button>
      <Son onChange={hanldeChange} />
    </div>
  )
  
}

从上面的案例可以看到 ,如果我们去改变value的值,那么对应的handleChange函数也就对应生成新的引用,那么这个时候Son组件也就会重新渲染,但是我们可以看到这里的Son组件只是想调用方法并不依赖父组件的展示数据,所以这里的重新渲染是没有必要的。

那么该如何处理这种情况呢,其实这里我们只需要保持handleChange的函数引用相同就行。

useMemoizedFn

js 复制代码
import { useMemo, useRef } from 'react';

type noop = (this: any, ...args: any[]) => any;

type PickFunction<T extends noop> = (
  this: ThisParameterType<T>,
  ...args: Parameters<T>
) => ReturnType<T>;

function useMemoizedFn<T extends noop>(fn: T) {
  const fnRef = useRef<T>(fn);

  fnRef.current = useMemo(() => fn, [fn]);

  const memoizedFn = useRef<PickFunction<T>>();
  if (!memoizedFn.current) {
    memoizedFn.current = function (this, ...args) {
      return fnRef.current.apply(this, args);
    };
  }

  return memoizedFn.current as T;
}

export default useMemoizedFn;

这里我们封装一个useMemoizedFn hook ,我们传入一个fn,把它存储在useRef中,当fn改变的时候做一个缓存,再创建一个memoizeFn来缓存我们的新建的方法,里面调用我们传入的函数,这里用单例设计模式第一次传的时候才会改变memoizefn的函数引用值,然后返回缓存函数。

这个时候我们回到页面上,可以看到我们改变value上的值的时候,Son组件没有重新渲染,符合预期结果。

缓存setValue

js 复制代码
const Son = React.memo(({ onChange }) => {
 
 return (
   <div onClick={() => onChange('加班到猝')}>
    我爱加班、我爱加班!
   </div>
 )
})

const Parent = () => {
  const [value, setValue] = useState('');

  
  return (
    <div>
      <p>喜提{ value }</p>
      <button onClick={() => setValue('加班')}>我爱加班</button>
      <Son onChange={setValue} />
    </div>
  )
  
}

上面案例中实际我们在父亲组件更改value的时候其实Son组件也重新加载了,这是因为更改value值导致组件更新对应的setValue的引用也发生了改变,所以造成了重复渲染,那么这里该如何处理呢。

js 复制代码
import { useCallback, useState } from 'react';

const useCallbackRefState = <T>() => {
  const [refState, setRefState] = useState<T | null>(null);
  const refCallback = useCallback((value: T) => setRefState(value), []);
  return [refState, refCallback];
};

export default useCallbackRefState;

这里我们使用一个简单的useCallback来缓存setValue方法,就可以到达Son组件不必要的渲染。

如何检测组件的更新的数据源

这里我推荐使用ahooks里面的一个很好用的hooks useWhyDidYouUpdate

js 复制代码
function Demo(props) {
  // 第一个参数 自定义一个组件名称 第二个参数就是想要检测的数据源
  useWhyDidYouUpdate('CustomCom', {
    ...props
  })
}

这个就能很好的检测哪些不必要的更新,来减少组件的重复渲染。

相关推荐
酷酷的阿云6 分钟前
不用ECharts!从0到1徒手撸一个Vue3柱状图
前端·javascript·vue.js
微信:137971205878 分钟前
web端手机录音
前端
齐 飞14 分钟前
MongoDB笔记01-概念与安装
前端·数据库·笔记·后端·mongodb
神仙别闹31 分钟前
基于tensorflow和flask的本地图片库web图片搜索引擎
前端·flask·tensorflow
sszmvb12341 小时前
测试开发 | 电商业务性能测试: Jmeter 参数化功能实现注册登录的数据驱动
jmeter·面试·职场和发展
测试杂货铺1 小时前
外包干了2年,快要废了。。
自动化测试·软件测试·python·功能测试·测试工具·面试·职场和发展
王佑辉1 小时前
【redis】redis缓存和数据库保证一致性的方案
redis·面试
真忒修斯之船1 小时前
大模型分布式训练并行技术(三)流水线并行
面试·llm·aigc
GIS程序媛—椰子1 小时前
【Vue 全家桶】7、Vue UI组件库(更新中)
前端·vue.js
DogEgg_0011 小时前
前端八股文(一)HTML 持续更新中。。。
前端·html