React组件到底什么时候render啊?

[[348213]]

为天心等地区用户提供了全套网页设计制作服务,及天心网站建设行业解决方案。主营业务为成都做网站、成都网站设计、天心网站设计,以传统方式定制建设网站,并提供域名空间备案等一条龙服务,秉承以专业、用心的态度为用户提供真诚的服务。我们深信只要达到每一位用户的要求,就会得到认可,从而选择与我们长期合作。这样,我们也可以走得更远!

今天我们React源码交流群里有个小伙伴提出个有趣的问题,觉得自己对React运行流程理解很到位的同学,可以来看看。

对于如下Demo,点击Parent组件的div,触发更新,Son组件会打印child render!么?

 
 
 
  1. function Son() { 
  2.   console.log('child render!'); 
  3.   return 
    Son
  4.  
  5.  
  6. function Parent(props) { 
  7.   const [count, setCount] = React.useState(0); 
  8.  
  9.   return ( 
  10.      {setCount(count + 1)}}> 
  11.       count:{count} 
  12.       {props.children} 
  13.     
 
  •   ); 
  •  
  •  
  • function App() { 
  •   return ( 
  •      
  •        
  •      
  •   ); 
  •  
  • const rootEl = document.querySelector("#root"); 
  • ReactDOM.render(, rootEl); 
  • 在线Demo地址[1]

     
     
     
    1. 右滑显示答案:                                                                                                                                                 不会     

    render需要满足的条件
    React创建Fiber树时,每个组件对应的fiber都是通过如下两个逻辑之一创建的:

    可以看到,当命中bailout逻辑时,是不会调用render函数的。

    所以,Son组件不会打印child render!是因为命中了bailout逻辑。

    bailout需要满足的条件
    什么情况下会进入bailout逻辑?当同时满足如下4个条件时:

    1. oldProps === newProps ?
    即本次更新的props(newProps)不等于上次更新的props(oldProps)。

    注意这里是全等比较。

    我们知道组件render会返回JSX,JSX是React.createElement的语法糖。

    所以render的返回结果实际上是React.createElement的执行结果,即一个包含props属性的对象。

    即使本次更新与上次更新props中每一项参数都没有变化,但是本次更新是React.createElement的执行结果,是一个全新的props引用,所以oldProps !== newProps。

    如果我们使用了PureComponent或Memo,那么在判断是进入render还是bailout时,不会判断oldProps与newProps是否全等,而是会对props内每个属性进行浅比较。

    2. context没有变化
    即context的value没有变化。

    3. workInProgress.type === current.type ?
    更新前后fiber.type是否变化,比如div是否变为p。

    4. !includesSomeLane(renderLanes, updateLanes) ?
    当前fiber上是否存在更新,如果存在那么更新的优先级是否和本次整棵fiber树调度的优先级一致?

    如果一致则进入render逻辑。

    就我们的Demo来说,Parent是整棵树中唯一能触发更新的组件(通过调用setCount)。

    所以Parent对应的fiber是唯一满足条件4的fiber。

    Demo的详细执行逻辑
    所以,Demo中Son进入bailout逻辑,一定是同时满足以上4个条件。我们一个个来看。

    条件2,Demo中没有用到context,满足。

    条件3,更新前后type都为Son对应的函数组件,满足。

    条件4,Son本身无法触发更新,满足。

    所以,重点是条件1。让我们详细来看下。

    本次更新开始时,Fiber树存在如下2个fiber:

     
     
     
    1. FiberRootNode 
    2.       | 
    3.   RootFiber     

    其中FiberRootNode是整个应用的根节点,RootFiber是调用ReactDOM.render创建的fiber。

    首先,RootFiber会进入bailout的逻辑,所以返回的App fiber和更新前是一致的。

     
     
     
    1. FiberRootNode 
    2.       | 
    3.   RootFiber       
    4.       | 
    5.   App fiber 

    由于App fiber是RootFiber走bailout逻辑返回的,所以对于App fiber,oldProps === newProps。并且bailout剩下3个条件也满足。

    所以App fiber也会走bailout逻辑,返回Parent fiber。

     
     
     
    1. FiberRootNode 
    2.       | 
    3.   RootFiber       
    4.       | 
    5.    App fiber 
    6.       | 
    7.  Parent fiber 

    由于更新是Parent fiber触发的,所以他不满足条件4,会走render的逻辑。

    接下来是关键

    如果render返回的Son是如下形式:

     
     
     
    1.  

    会编译为

     
     
     
    1. React.createElement(Son, null) 

    执行后返回JSX。

    由于props的引用改变,oldProps !== newProps。会走render逻辑。

    但是在Demo中Son是如下形式:

     
     
     
    1. {props.children} 

    其中,props.children是Son对应的JSX,而这里的props是App fiber走bailout逻辑后返回的。

    所以Son对应的JSX与上次更新时一致,JSX中保存的props也就一致,满足条件1。

    可以看到,Son满足bailout的所有条件,所以不会render。

    总结
    当你理解这4个条件后,对于React组件更新会有全新的认识。

    不得不说,React真是太难了,打工人流下了不争气的眼泪。

     

    当前名称:React组件到底什么时候render啊?
    链接地址:http://www.shufengxianlan.com/qtweb/news31/62031.html

    网站建设、网络推广公司-创新互联,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

    广告

    声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联

    猜你还喜欢下面的内容

    网站设计知识

    行业网站建设