函数组件在使用Hook时的性能优化

开篇

列出渲染次数多的原因

  1. 回调中有异步操作(减少回调中的异步操作)。

  2. 传递给子组件的props在父组件渲染时被重新定义(使用useCallback、useMemo配合React.memo来减少子组件的渲染)。

使用useCallback时

  1. 当作为props传递给子组件时,配合React.memo用于减少子组件的渲染次数。

  2. 在当前组件中只是减少了重新定义的次数,重新定义的性能损耗其实可以忽略。

  3. 可通过setState回调函数,来减少依赖项,从而减少props被重新定义的次数。

使用useMemo时

  1. 其实只是提供了someRef.current的快捷使用方式,缓存了变量的索引。
  2. 在当前组件使用时,用于缓存通过较大计算量得到的值。
  3. 当作为props传递给子组件时,配合React.memo用于减少子组件的渲染次数。

实例

回调中有异步操作

javascript 复制代码
const Child = ({handleClick})=>{
    console.log('child: 我渲染了')

    return (
        <button onClick={handleClick}>点击我</button>
    )
}
const App = ()=>{
    const [num,setNum] = useState(0)

    const handleClick = ()=>{
        
        setNum(num + 1)
        setNum(num + 1)
        setNum(num + 1)
        console.log('num',num)

        setTimeout(()=>{
            console.log(num)
            setNum((a)=>{
                console.log(a)
                return a + 1
            })
        })
    }
    console.log('父:渲染了')
    return (
        <>
            {num}
            <br/>
            <Child handleClick={handleClick}/>
        </>
    )

}

export default App;

对于setTimeout中console.log(num)输出0的理解:

setTimeout的回调产生了闭包,冻结了num的初始值。

javascript 复制代码
const Child = ({handleClick})=>{
    console.log('child: 我渲染了')

    return (
        <button onClick={handleClick}>点击我</button>
    )
}
const App = ()=>{
    const [num,setNum] = useState(0)

    const handleClick = ()=>{
        
        setNum(num + 1)
        setNum(num + 1)
        setNum(num + 1)
        console.log('num',num)
        // 排除setTimeout影响,研究闭包的形成
        ;(()=>{
            console.log(num)
            setNum((a)=>{
                console.log(a)
                return a + 1
            })
        })()
    }
    console.log('父:渲染了')
    return (
        <>
            {num}
            <br/>
            <Child handleClick={handleClick}/>
        </>
    )

}

export default App;

消除影响,减少次数

如果回调中有异步操作,并且异步操作中也需要改变组件状态,就应该把异步操作外改变状态的语句移到异步操作内执行。

注:实践中发现,此种方式增加的渲染次数无法减少,因为异步中的set是同步执行的。从逻辑讲,确实两个setState的时机是不同的,异步中的setState需要等待,所以优先渲染已准备好的state,不让用户等待,是合理的。

思:一般遇到此种情况,为减少渲染次数,分拆成更小的组件去渲染可以解决问题,提升性能;或在不同时机去执行set操作。

传递给子组件的props在父组件渲染时被重新定义

javascript 复制代码
const Child = ({handleClick})=>{
    console.log('child: 我渲染了')

    return (
        <button onClick={handleClick}>点击我</button>
    )
}

const ChildTwo = React.memo(({ handleClick2 }) => {
    console.log('childtwo: 我渲染了')
    return (
        <button onClick={handleClick2}>点击我2</button>
    )
})

const App = ()=>{
    const [num,setNum] = useState(0)

    const handleClick = ()=>{
        setNum(num+1)
    }
    // const handleClick2 = useCallback(()=>{
    //     setNum(1)
    // },[])
    // const handleClick2 = useCallback(()=>{
    //     setNum(num+1)
    // },[num])
    const handleClick2 = useCallback(()=>{
        setNum(num=>num+1)
    },[])
    const obj = useMemo(()=>{
        return {}
    },[])
    console.log('父:渲染了')

    return (
        <>
            {num}
            <br/>
            <Child handleClick={handleClick}/>
            <ChildTwo handleClick2={handleClick2}/>
            {/* <ChildTwo a={obj}/> */}
        </>
    )

}

export default App;

注:

useCallback和useMemo在使用时的依赖记得要加上,因为他们两个是根据依赖的变化来重新定义的,不然我们在其内部使用num时,是不能拿到num的最新值的,除非你使用setNum(num=>num+1),这也是减少依赖的好方法,不过不适用所有场景。

引申内容

  • 自执行函数前方必须加上分号,否则js解释器会因为无法识别而报错。

  • react没有暴露主动控制组件重新渲染的接口,不过可以自己为组件传入一个props,通过改变这个props来实现组件的重新渲染。

  • 纯展示列表不会出现数组变更的,可以用index作为key;数组可变更的列表就需要使用id等唯一且不变的标识作为key。

后端如何没有传递id,初始的时候自己可对数组格式化��入id。

可设置全局变量,做自增操作设置id,以保证id的唯一性。

相关推荐
软件技术NINI1 分钟前
泉州html+css 4页
前端·javascript·css·html
再吃一根胡萝卜2 分钟前
OpenScreen:免费开源的录屏神器,做出专业级演示视频
前端
Cloud_Shy6184 分钟前
Python 数据分析基础入门:《Excel Python:飞速搞定数据分析与处理》学习笔记系列(第十一章 Python 包跟踪器 下篇)
前端·后端·python·数据分析·excel
kyriewen5 分钟前
我用AI把公司10万行代码屎山重构了,CTO看了代码后说:你提前转正
前端·javascript·ai编程
ttwuai8 分钟前
XYGo Admin 菜单与路由:Vue3 动态路由 + GoFrame 权限菜单的完整实现方案
前端·vue·后台框架
程序员码歌16 分钟前
OpenSpec 到 Superpowers:AI 编码从说清到做对
android·前端·人工智能
爱编程的小新☆16 分钟前
LangGraph4j工作流框架
前端·数据库·ai·langchain·langgraph4j
@PHARAOH32 分钟前
HOW - 构建一个轻量前后端一体服务
前端·微服务·服务端
无限进步_43 分钟前
【C++】C++11的类功能增强与STL变化
java·前端·数据结构·c++·后端·算法