Bootstrap

【Vue3笔记】Ref和Reactive

参考:

学习Vue3 第六章(认识Ref全家桶)

学习Vue3 第七章(认识Reactive全家桶)

Ref和Reactive关系

Ref会在一定条件下调用reactive,reactive会使用proxy代理输出响应式对象。

因此,响应式其实是由Proxy代理实现的,这也是Vue3的响应式原理。注意,Vue2不一样,采用 Object.defineProperty()的get()与set()来实现响应式。

Ref支持基本数据类型和引用数据类型,reactive只支持引用数据类型。

详情见vue源码https://github.com/vuejs/core,

响应式代码位于packages>reactivity>src目录下。

Ref

  1. Ref(),返回一个对象,使用.value获取值。如果不使用ref,v-model绑定的输入框属性改变后,相应data不会相应改变
  2. isRef:判断是否是ref对象,
  3. shallowRef:浅层次响应式,ref.value是响应式,.value之后的属性不是响应式。即只有ref.value赋值能触发视图更新,ref.value.any能赋值但不触发视图更新。。
  4. triggerRef: 触发ref和shallowRef的视图更新
  5. customRef:自定义ref,get和set函数,在set中可以加防抖
  6. Ref调用 triggerRef,所以ref和shallowRef不能一起使用,ref的triggerRef会影响shallowRef,使得shallowRef的ref.value.any方法能够触发视图更新
  7. 元素行内设置ref属性,再使用属性名可以获取到dom元素。相当于Vue2的this.$refs方法。

Ref源码

  1. 先调用createRef(),ref和shallowRef区别在于createRef()第二个参数__V_isShallow。__V_isShallow =false为ref,__V_isShallow =true为shallowRef。
  2. createRef()中,返回RefImpl类。
  3. RefImpl类中,若__V_isShallow =true,那么要生成shallowRef对象则,调用toRaw(),直接返回值。若__V_isShallow =false那么要生成Ref对象。调用toReactive()
  4. toReactive()中,如果传入基本数据类型,直接返回值,如果传入引用数据类型,会调用reactive()(见reactive源码)
  5. 完成RefImpl类创建后。不管是创建的对象是Ref还是shallowRef都会调用trackRefValue()triggerRefValue()。trackRefValue()收集数据,triggerRefValue()触发视图更新,而 triggerRefValue()又会调用triggerEffect。由于最终经过triggerRefValue(),所以ref和shallowRef在单文件中不能同时出现。

Reactive

  1. Reactive(),返回一个对象,使用.any获取值。
  2. isReactive:判断是否是Reactive对象,
  3. Readonly:设置为只读。Readonly过的原始数据无法直接修改,但对Readonly过的reactive原始数据赋值,能够修改

                                            

  1. shallowReactive:浅层次响应式Reactive.any是响应式,.any.any以及更深层的属性不是响应式。即只有Reactive赋值能触发视图更新,Reactive.any能赋值但不触发视图更新。
  2. Reactive和shallowReactive不能一起使用,Reactive会影响shallowReactive,使得shallowReactive能够触发视图更新

ref和Reactive触发视图更新时,会对整个template(即App.vue)的所有组件进行re-render再渲染(diff算法比较数据)。因此shallowReactive和shallowRef的数据也会被渲染。

Reactive源码

使用proxy劫持属性,完成响应式

CreateReactiveObject()创建reactive对象

通过WeakMap实现弱引用,对象不使用后,键值对会自动销毁。

toRef、toRefs、toRaw

ToRef

toRef创建一个ref对象,使其指向被包含的数据(类似指针访问地址),更改toRef数值,也会更改reactive数值。

对于非响应式对象,只修改属性不更新视图,对于响应式对象才能更新视图。

toRef主要用于从原reactive对象解构部分属性,转换成响应式对象(普通解构会丧失响应式)。

应用场景:比如某个函数需要reactive对象部分属性作为参数,又要保持响应式,那就用toRef包裹

ToRefs

从reactive对象中解构每个属性为单一的ref对象,如果不使用toRefs,那么会解构为普通对象,修改数值不会更新视图。

ToRaw

将响应式对象变为普通对象,即取消Proxy代理

;