react useCallback应用

函数组件

函数组件

函数组件实际上就是一个函数,接收props对象参数,返回一个React元素。

js 复制代码
function Welcome(props) {
  return <h1>Hello, {props.name}</h1>;
}

函数组件重新渲染

React 组件在组件的状态 state 或者组件的属性 props 改变的时候,会重新渲染。函数组件重新渲染时,会重新执行这个组件函数。

useCallback用法

js 复制代码
let memoizedCallback = useCallback(fn, dependencies)

参数

fn:

想要缓存的函数。在初次调用(初次渲染)时,返回函数fn。下一次调用(下一次渲染)时,如果依赖项没有变化,返回上次的函数fn;如果依赖项发生了变化,返回最新的函数fn'
dependencies:

是否更新fn的依赖列表。响应式值包括 propsstate,和所有在你组件内部直接声明的变量和函数。

返回值

在初次渲染时,useCallback 返回你已经传入的 fn

在之后的渲染中, 如果依赖没有改变,useCallback 返回上一次渲染中缓存的 fn 函数;否则返回这一次渲染传入的 fn

场景

减少子组件重新渲染

当父组件重新渲染时,子组件也会重新渲染,这样造成不必要的开销。React.memo解决了这个问题。React.memo会浅比较当前组件的props与上一次渲染时的 props。如果props没有变化,跳过重新渲染。

当父组件的函数通过props传递给子组件时,使用useCalback包裹函数,配合React.memo防止子组件重新渲染。

js 复制代码
// 子组件使用 React.memo 优化
const ChildComponent = React.memo(function ChildComponent({ onClick }) {
  console.log("ChildComponent rendered");
  return <button onClick={onClick}>Click me</button>;
});

// 父组件使用 useCallback
function ParentComponent() {
  const [count, setCount] = useState(0);
  const [text, setText] = useState("");

  // 使用 useCallback 记忆回调函数
  const handleClick = useCallback(() => {
    setCount(c => c + 1);
  }, []); // 空依赖数组,因为不依赖任何值

  return (
    <div>
      <input value={text} onChange={e => setText(e.target.value)} />
      <p>Count: {count}</p>
      <ChildComponent onClick={handleClick} />
    </div>
  );
}

防止频繁触发useEffect

如下代码,父组件的函数handleChange函数传递给子组件。当在子组件中的useEffect hook中调调用父组件handleChange函数时,会导致无限循环渲染父子组件。

子组件useEffect中调用handleChange函数后,执行这个函数。函数中有语句setState,触发父组件重新渲染。父组件重新渲染,在重新渲染时创建新的handleChange函数。
handleChange函数改变了,则又会执行useEffect里面的handleChange函数。这样造成了无限循环调用。

js 复制代码
function Parent(){
  const [state, setState] = useState()
  
  function handleChange(input){
    setState(...)
  }

  return <Child onChange={handleChange} />
}

function Child({onChange}){
  const [state, setState] = useState()
  
  useEffect(()=>{
    onChange(...)
  },[onChange])

  return "Child"
}

下面使用useCallback解决这一问题。给父组件的handleChange函数使用useCallback包裹。

js 复制代码
function Parent(){
  const [state, setState] = useState()
  
  let handleChange = useCallback((input) => {
    setState(...)
  },[])

  return <Child onChange={handleChange} />
}

function Child({onChange}){
  const [state, setState] = useState()
  
  useEffect(()=>{
    onChange(...)
  },[onChange])

  return "Child"
}
相关推荐
广州华水科技10 小时前
单北斗GNSS形变监测系统在桥梁安全中的应用与技术解析
前端
打小就很皮...11 小时前
ShowCountCard 功能迭代:新增周月对比属性,完善数据可视化场景
前端·react.js·信息可视化
IT_陈寒11 小时前
Redis性能翻倍的7个冷门技巧:从P5到P8都在偷偷用的优化策略!
前端·人工智能·后端
Moonbit11 小时前
MoonBit Meetup 丨 手把手带你走进 AI 编程新世代
前端·后端·程序员
携欢11 小时前
PortSwigger靶场之 CSRF where token is not tied to user session通关秘籍
前端·csrf
HHHHHY12 小时前
使用阿里lowcode,手搓一个Carousel 走马灯容器组件
前端·react.js
用户3521201956012 小时前
React-router v7
前端
Mintopia12 小时前
⚡ AI 时代,全栈 Next.js 开发的激情在哪里?
前端·aigc·全栈
Hello123网站12 小时前
300多个Html5小游戏列表和下载地址
前端·html·html5
Stringzhua12 小时前
ElementUi【饿了么ui】
前端·ui·elementui