联系作者
联系作者

# 5.1 React Virtual DOM原理

# 5.1.1 Virtual DOM是什么?

官方定义:virtual DOM是一种编程理念(数据驱动视图),将ui虚拟的保持到内存中,并且通过某些库渲染成真实的dom,这个过程又叫做协调。

总结:virtual dom是一种编程理念,将ui节点抽象成js对象。

# ui节点抽象

Learn Once,Write anywhere:因为提供了对HTML DOM的抽象,所以在web开发中,通常不需要去调用DOM API,也是因为抽象,所以React也可以开发Native(React Native)

# Virtual dom 构建UI

以我们经常见的web开发为例,来看下React是怎么通过Virtual DOM渲染成HTML元素的。

    class App extends Component {
        state={
            text:'Virtual DOM'
        }
        render () {
            const {text}=this.state
            return (
                <div>
                    {text}
                <div>
            )
        }
    }
1
2
3
4
5
6
7
8
9
10
11
12
13

可以看到React是通过render方法渲染Virtual DOM ,从而绘制出真实DOM。意味着每次修改state的值就会执行render方法

# 5.1.2 Virtual DOM VS 原生DOM

  • 原生DOM更新

DOM API 调用更新UI

  • Virtual DOM更新
  1. 每次render都会产生一份新的’react dom‘
  2. Virtual DOM要对新旧‘react dom’进行比较,从而确定在旧‘dom’的基础上进行多少变更
  3. 确定最优的变更策略之后调用DOM API更新

Virtual DOM数据结构

    const globaldom={
        tagname:'html',
        children:[{
            tagname:'head'
        },{
            tagname:'body',
            children:[
                {
                  tagName:'div',
                  attributes:{
                      className:'test'
                  }
                }
            ]
         }
        ]
    }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17

对UI节点抽象

在Virtual DOM中,对HTML节点进行抽象,用js对象的形式表示HTML

改变了过去对HTML节点的理解

呈现在用户面前的页面就是一个复杂的递归对象

# 5.1.3 Virtual DOM Diff

  • Virtual DOM如何提搞性能
  1. 我们将render产生的Virtual DOM简称 ‘Vdom’
  2. 通常调用setState方法触发Vdom更新
  3. 通过对比新旧‘Vdom’确定最优实现新的‘vdom’所需的操作
  • Virtual DOM Diff的层次
  1. 层级级别的比较
  2. 元素级别的比较

# Component Diff

假设将图中的D节点更换成G节点,会把D节点删掉,然后创建G节点;只会同层级比较,这样会降低diff的复杂度,大概率下面的层级不一样的。

# Element Diff

  1. createChild
  2. moveChild
  3. removeChild

我们元素之间的对比中,会有3种情况:创建节点、删除节点、移动节点

  • 创建节点
    createChild:function(child,afterNode,mountImage){
        return makeInserMarku(mountImage,afterNode,child._mountIndex)
    }
1
2
3
  • 删除节点
    removeChild:function(child,node){
        return makeRemove(child,node)
    }
1
2
3
  • 移动节点

假设我们把节点3移动到第一的位置

// 老的节点和新的节点相同,说明需要移动节点
if(preChild===nextChild){
    updates=enqueue(updates,this.moveChild(preChild,lastPlaceNode,nextIndex,lastIndex))
    lastIndex=Math.max(preChild._mountIndex,lastIndex)
    preChild._mountIndex=nextIndex
}
// 移动的方法
moveChild:function (child,afterNode,toIndex,lastIndex){
    // _mountIndex是这个元素在原有节点的顺序,lastIndex是这次要更新的顺序
    if(child._mountIndex<lastIndex){
        return makeMove(child,afterNode,toIndex)
    }
}
1
2
3
4
5
6
7
8
9
10
11
12
13

在实际中最后一个节点移动到第一位置的时候,不是把最后一个节点直接移动到第一的位置,而是把这个节点前面的节点依次往后移动,直到最后一个节点成为第一的位置。

# 5.1.3 扩展资料

Virtual DOM 定义

Virtual DOM Node

VDom与 DOM 的区别

React性能优化:Virtual Dom原理浅析

[译] Virtual Dom 和 Diff 算法

# 5.2 React Reconciler(react协调的过程)

# 5.2.1 React Reconclier(15版本)

# 5.3 new componentLifecyle(新生命周期)

# 5.3.1 React组件新生命周期详解

老的生命周期

react 16.3 增加了getDeriveStateFromProps和getSnapshotBeforeUpdate 2个静态方法

react 16.4中增加了在setState和forceUpdate()都会执行这个静态方法getDeriveStateFromProps

# 5.3.2 挂载阶段

  1. 挂载阶段的函数
  • constructor 构造函数,初始化state,以及为事件处理函数绑定实例
  • getDeriveStateFromProps 新增的静态方法,返回一个新的state,或者是null,静态方法里的,这里面的this是指向全局变量
  • render 渲染函数
  • componentDidMount 挂载成功后立即调用
  1. 更新阶段的函数
  • getDeriveStateFromProps props变化或者state方法触发
  • shouldComponentUpdate 判断是否进行更新
  • render 函数
  • getSnapshotBeforeUpdate 生命周期方法在更新之前(如:更新 DOM 之前)被调用。此生命周期的返回值将作为第三个参数传递给 componentDidUpdate。(通常不需要,但在重新渲染过程中手动保留滚动位置等情况下非常有用。)
  • componentDidUpdate 更新后会被立即调用
  1. 卸载阶段的函数
  • componentWillUnmount 卸载函数,组件卸载及销毁之前直接调用,主要用于清除一些在组件生命周期订阅,真实DOM事件以及setTimeout/setinterval的返回值

# 异常捕获

  • componentDidCatch 生命周期方法在后代组件抛出错误后被调用,方法接受两个参数(err,info),分别是错误信息和错误组件的栈信息
  • getDerivedStateFromError 在后代组件抛出错误后调用,接受一个参数(error)表示具体的错误信息

# 5.3.3 新版组件升级

  • componetWillMount (17.0被移除)

    • render 方法之前调用,在此调用setState并不会触发再次渲染
    • 通常会在这个方法中进行页面标题的一些修改以及其他与再次render不相关的操作
  • UNSAFE_componentWillMount (16.3 新增不安全的生命周期引入别名)

    • 与state相关的操作挪到constructor方法中执行
    • 异步操作挪到componentDidMount中执行
  • componentWillUpdate(17.0被移除)

    • 在组件收到新的props或state时,会在渲染之前调用
    • 方法内不能调用setState,触发循环,内存泄漏
  • UNSAFE_componentWillUpdate(16.3 新增不安全的生命周期引入别名)

    • 应该在shouldComponentUpdate
  • componentWillReceiveProps(17.0被移除)

    • 接受父级传递过来最新的props,转化为组件内的state
    • 判断是否进行更新或者执行异步请求数据
  • UNSADFE_componentWillReceiveProps(16.3 新增不安全的生命周期引入别名)

    • 与渲染相关的props直接渲染,不需要处理为组件内的state
    • 异步数据请求在componentDidUpdate中处理
    • getDerivedStateFromPropss方法替换,需要考虑生命周期的执行顺序
上次更新: 8/7/2024, 2:24:53 AM

© 2024 web全栈体系. All Rights Reserved.(所有文章未经授权禁止转载、摘编、复制或建立镜像,如有违反,追究法律责任。)

豫ICP备19041317号-1