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

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

相关推荐
漂流瓶jz4 小时前
总结CSS组件化演进之路:命名规范/CSS Modules/CSS in JS/原子化CSS
前端·javascript·css
踩着两条虫5 小时前
「AI + 低代码」的可视化设计器
开发语言·前端·低代码·设计模式·架构
Jagger_5 小时前
项目上线忙碌结束之后,为什么总想找点事做?
前端
GalenZhang8885 小时前
OpenClaw 配置多个飞书账号实战指南
前端·chrome·飞书·openclaw
萌新小码农‍6 小时前
python装饰器
开发语言·前端·python
Wilber的技术分享7 小时前
【大模型面试八股 3】大模型微调技术:LoRA、QLoRA等
人工智能·深度学习·面试·lora·peft·qlora·大模型微调
threelab7 小时前
Three.js 初中数学函数可视化 | 三维可视化 / AI 提示词
开发语言·前端·javascript·人工智能·3d·着色器
爱学习的程序媛7 小时前
浏览器工作原理全景解析
前端·浏览器·web
我是若尘8 小时前
用 Git Worktree 同时开多个需求,不用来回 stash
前端
IT_陈寒8 小时前
Vue的v-for为什么不加key也能工作?我差点翻车
前端·人工智能·后端