React@16.x(27)useCallBack

1,引入

先来看一个例子:

1.1,举例:

import React, { PureComponent, useState } from "react";

class Child extends PureComponent {
    render() {
        console.log("child render");
        return (
            <>
                <div>{this.props.txt}</div>
                <button onClick={this.props.onClick}>改变txt</button>
            </>
        );
    }
}

export default function App() {
    console.log("App render");
    const [txt, setTxt] = useState("abc");
    return (
        <Child
            txt={txt}
            onClick={() => {
                setTxt(Math.random());
            }}
        ></Child>
    );
}

注意,子组件是类组件,并且使用了 PureComponent,所以每次 props 改变,也就是点击按钮都会输出:

App render
child render

现在修改父组件,增加了一个状态的改变:

export default function App() {
    console.log("App render");
    const [txt, setTxt] = useState("abc");
    const [n, setN] = useState(0);
    return (
        <>
            <Child
                txt={txt}
                onClick={() => {
                    setTxt(Math.random());
                }}
            ></Child>
            <input
                type="number"
                value={n}
                onChange={(e) => {
                    setN(e.target.value);
                }}
            />
        </>
    );
}

理论上来说,状态变量 n 发生变化时,父组件会重新渲染打印 App render 没有问题。但子组件的不会有打印,因为传递给它的状态没有改变

可状态变量 n 发生变化时,会打印:

App render
child render

其实问题出现在 this.props.onClick 上了,注意到传递给它的是一个函数,而在父组件重新渲染时,会产生一个新的函数
此时对子组件来说,函数的引用地址发生了变化,所以 PureComponent “失效了”。

<Child
   txt={txt}
    onClick={() => {
        setTxt(Math.random());
    }}
></Child>

1.2,新的问题

接着会发现,无论怎么做,传递给子组件的都是一个新的函数:

写法1:

export default function App() {
    // ...
    // App 组件重新调用,会重新生成新函数。
    function handleClick() {
        setTxt(Math.random());
    }
    return (
        <>
            <Child txt={txt} onClick={handleClick}></Child>
            { // ... }
        </>
    );
}

写法2:

每次还是新的函数。

function handleClick(setTxt) {
    setTxt(Math.random());
}

export default function App() {
    // ...
    return (
        <>
            <Child txt={txt} onClick={() => handleClick(setTxt)}></Child>
            { // ... }
        </>
    );
}

所以,需要一个方法,能够将函数的引用值固定,不要每次渲染都是新函数,才能解决这个问题。

2,useCallBack 介绍

作用:用于得到一个固定引用值的函数。通常用它来进行性能优化。

使用:接收2个参数,

  1. 参数1函数,useCallBack 会固定该函数的引用,只要依赖项(参数2)没有发生变化,则使用返回之前的函数地址。
  2. 参数2数组,依赖项。
  3. 返回值,固定函数的引用地址。

可以看到,和 useEffect 很像。依赖项不发生变化,第1个函数不会再次执行。

改造上面的例子

import React, { useCallback } from "react";

export default function App() {
    // ...
    const handleClick = useCallback(() => {
        setTxt(Math.random());
    }, []);
    return (
        <>
            <Child txt={txt} onClick={handleClick}></Child>
            { // ... }
        </>
    );
}

第2个参数传不传 [txt] 效果是一样的。
因为如果传递 txt,则只有 txt 发生变化,才会返回新函数。


以上。

相关推荐

  1. React@16.x27useCallBack

    2024-06-14 07:48:03       9 阅读
  2. React@16.x23)useEffect

    2024-06-14 07:48:03       12 阅读
  3. React@16.x26)useContext

    2024-06-14 07:48:03       9 阅读
  4. React@16.x28)useMemo

    2024-06-14 07:48:03       8 阅读
  5. React@16.x25)useReducer

    2024-06-14 07:48:03       11 阅读
  6. React@16.x24)自定义HOOK

    2024-06-14 07:48:03       10 阅读

最近更新

  1. TCP协议是安全的吗?

    2024-06-14 07:48:03       16 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-06-14 07:48:03       16 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-06-14 07:48:03       15 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-06-14 07:48:03       18 阅读

热门阅读

  1. 深入理解服务器路由功能:配置与应用详解

    2024-06-14 07:48:03       5 阅读
  2. TCP是什么、UDP是什么,它们有什么区别

    2024-06-14 07:48:03       6 阅读
  3. WHAT - React 学习系列(一)

    2024-06-14 07:48:03       9 阅读
  4. .NET C# 实现国密算法加解密

    2024-06-14 07:48:03       3 阅读
  5. VB.net与C# 调用InitializeComponent的区别

    2024-06-14 07:48:03       6 阅读
  6. HarmonyOS(35) @State使用注意事项

    2024-06-14 07:48:03       5 阅读
  7. php学习笔记

    2024-06-14 07:48:03       6 阅读