项目性能优化 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
  })
}

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

相关推荐
这个需求建议不做几秒前
vue3打包配置 vite、router、nginx配置
前端·nginx·vue
QGC二次开发5 分钟前
Vue3 : Pinia的性质与作用
前端·javascript·vue.js·typescript·前端框架·vue
云草桑17 分钟前
逆向工程 反编译 C# net core
前端·c#·反编译·逆向工程
布丁椰奶冻22 分钟前
解决使用nvm管理node版本时提示npm下载失败的问题
前端·npm·node.js
AntDreamer36 分钟前
在实际开发中,如何根据项目需求调整 RecyclerView 的缓存策略?
android·java·缓存·面试·性能优化·kotlin
Leyla1 小时前
【代码重构】好的重构与坏的重构
前端
影子落人间1 小时前
已解决npm ERR! request to https://registry.npm.taobao.org/@vant%2farea-data failed
前端·npm·node.js
世俗ˊ1 小时前
CSS入门笔记
前端·css·笔记
子非鱼9211 小时前
【前端】ES6:Set与Map
前端·javascript·es6
6230_1 小时前
git使用“保姆级”教程1——简介及配置项设置
前端·git·学习·html·web3·学习方法·改行学it