Vue3 中 setup,ref 和 reactive 的理解

setup

Vue3中使用了Composition API这种写法,使得所有的组合API函数都在此使用, 只在初始化时执行一次。
函数如果返回对象, 对象中的属性或方法, 模板中可以直接使用

setup的一些细节问题

1,setup执行的时机 和 返回值:

  • 在beforeCreate之前执行(一次), 此时组件对象还没有创建,不能通过this来访问data/computed/methods / props
  • 返回值一般为一个对象,返回对象中的属性/方法,会与data/methods函数返回对象的属性合并成为组件对象的属性/方法
  • methods中可以访问setup提供的属性和方法, 但在setup方法中不能访问data和methods;
  • setup不能是一个async函数: 因为返回值不再是return的对象, 而是promise, 模板看不到return对象中的属性数据
  • 如果有重名, setup优先

2,setup的参数:

  • setup(props, context) / setup(props, {attrs, slots, emit})
  • props: 包含props配置声明且传入了的所有属性的对象
  • attrs: 包含没有在props配置中声明的属性的对象, 相当于 this.$attrs
  • slots: 包含所有传入的插槽内容的对象, 相当于 this.$slots
  • emit: 用来分发自定义事件的函数, 相当于 this.$emit
props: ['msg'],
emits: ['fn'], // 可选的, 声明了更利于程序员阅读, 且可以对分发的事件数据进行校验
setup (props, {
     
     attrs, emit, slots}) {
    
    
	console.log(props.msg, attrs.msg2, slots, emit)
	function toClick(){
    
    
		emit(fn, 'xxxxx') // 相当于 this.$emit
	}
}

ref

作用:定义一个数据的响应式
语法:const xxx = ref(initValue)
一般用来定义一个基本类型的响应式数据

  • 创建一个包含响应式数据的引用(reference)对象
  • js中操作数据: xxx.value
  • 模板中操作数据: 不需要.value

结合setup 和 ref 使用说明

其中vue2的写法为:

<template>
  <h2>{
   
   {count}}</h2>
  <hr>
  <button @click="update">更新</button>
</template>

<script>
export default {
      
      
  /* 在Vue3中依然可以使用data和methods配置, 但建议使用其新语法实现 */
   data () {
      
      
     return {
      
      
       count: 0
     }
   },
   methods: {
      
      
     update () {
      
      
       this.count++
     }
   }
}
</script>

vue3的写法为:

  1. setup最终会返回一个对象,并且对象中的值可以在模板中使用,如count
  2. 保证响应式,必须使用ref
<template>
  <h2>{
   
   {count}}</h2>
  <hr>
  <button @click="update">更新</button>
</template>

<script>
import {
      
      
  ref
} from 'vue'
export default {
      
      
  /* 使用vue3的composition API */
  setup () {
      
      
    // 定义响应式数据 ref对象
    const count = ref(1)
    console.log(count)

    // 更新响应式数据的函数
    function update () {
      
      
      count.value = count.value + 1
    }

    return {
      
      
      count,
      update
    }
  }
}
</script>

reactive

作用:定义多个数据的响应式,例如一个对象;
const isProxy = reactive(obj): 接收一个普通对象然后返回该普通对象的响应式代理器对象
响应式转换是“深层的”:会影响对象内部所有嵌套的属性
内部基于 ES6 的 Proxy 实现,通过代理对象操作源对象内部数据都是响应式的

<template>
  <h2>name: {
   
   {state.name}}</h2>
  <h2>age: {
   
   {state.age}}</h2>
  <h2>wife: {
   
   {state.wife}}</h2>
  <hr>
  <button @click="update">更新</button>
</template>

<script>
import {
      
       reactive } from 'vue'
export default {
      
      
  setup () {
      
      
    /*  定义响应式数据对象 */
    let obj = {
      
      
      name: 'tom',
      age: 25,
      wife: {
      
      
        name: 'marry',
        age: 22
      },
    };
    // 此时state通过 reactive代理了obj,使其内属性成为响应式的;
    const state = reactive(obj)
    // 此时打印state 会得到一个Proxy的一个对象
    // state---代理对象,obj---目标对象
    console.log(state)

    const update = () => {
      
      
      state.name += '--'
      state.age += 1
      state.wife.name += '++'
      state.wife.age += 2
    }

    return {
      
      
      state,
      update,
    }
  }
}
</script>

总结:如果操作代理对象,目标对象中的值也会跟着改变,如果想要页面跟着渲染,也是操作代理对象;

ref 和 reactive的一些细节

  • 是Vue3的 composition API中2个最重要的响应式API;
  • ref用来处理基本类型数据, reactive用来处理对象(递归深度响应式);
  • 如果用ref对象/数组, 内部会自动将对象/数组转换为reactive的代理对象;
  • ref内部: 通过给value属性添加getter/setter来实现对数据的劫持;
  • reactive内部: 通过使用Proxy来实现对对象内部所有数据的劫持, 并通过Reflect操作对象内部数据;
  • ref的数据操作: 在js中要.value, 在模板中不需要(内部解析模板时会自动添加.value)
export default {
    
    

  setup () {
    
    
    const m1 = ref('abc')
    const m2 = reactive({
    
    x: 1, y: {
    
    z: 'abc'}})

    // 使用ref处理对象  ==> 对象会被自动reactive为proxy对象
    const m3 = ref({
    
    a1: 2, a2: {
    
    a3: 'abc'}})
    console.log(m1, m2, m3)
    console.log(m3.value.a2) // 也是一个proxy对象

    function update() {
    
    
      m1.value += '--'
      m2.x += 1
      m2.y.z += '++'

      m3.value = {
    
    a1: 3, a2: {
    
    a3: 'abc---'}}
      m3.value.a2.a3 += '==' // reactive对对象进行了深度数据劫持
      console.log(m3.value.a2)
    }

    return {
    
    
      m1,
      m2,
      m3,
      update
    }
  }
}

vue3响应式虽然写起来麻烦,但是减少了vue2响应式带来的性能开销

猜你喜欢

转载自blog.csdn.net/weixin_44684357/article/details/132177322