Vue 3学习一:介绍和常用的API

目录

1. 介绍

2. 常用 Composition API

2.1 setup

2.2 ref函数

2.3 reactive函数

2.4 Vue3中的响应式原理

2.4.1 vue2的响应式

2.4.2  Vue3.0的响应式

 2.5 reactive对比ref

2.6 setup的两个注意点

2.7 computed函数

 2.8 watch函数

2.9 watchEffect函数

 2.10 vue 3 生命周期

 2.11 自定义hook函数

2.12 toRef 和 toRefs


1. 介绍

2020年9月18日,Vue.js发布3.0版本,代号:One Piece(海贼王)

vue 3 的优点

  • 性能的提升

  • 源码的升级

  • 拥抱TypeScript
  • 新的特性

import { createApp } from 'vue', 引入的不是构造函数,是一个工厂函数

vue3 中的模板可以没有根标签

2. 常用 Composition API

2.1 setup

  • Vue3.0中一个新的配置项,值为一个函数。
  • setup是所有Composition API(组合API)
  • 组件中所用到的:数据、方法等等,均要配置在setup中。
  • 使用时模板和2同,调用是不用this,直接写xxx.value
  • setup函数的两种返回值:
  1. 若返回一个对象,则对象中的属性、方法, 在模板中均可以直接使用。(重点关注!)

  2. 若返回一个渲染函数:则可以自定义渲染内容。(了解)

注意点:

  1. 建议不要与Vue2.x配置混用

    • Vue2.x配置(data、methos、computed...)中可以访问到setup中的属性、方法。

    • 但在setup中不能访问到Vue2.x配置(data、methos、computed...)。

    • 如果有重名, setup优先。

  2. setup不能是一个async函数,因为返回值不再是return的对象, 而是promise, 模板看不到return对象中的属性。(后期也可以返回一个Promise实例,但需要Suspense和异步组件的配合)

2.2 ref函数

  • 作用: 定义一个响应式的数据

  • 语法: const xxx = ref(initValue)

    • 创建一个包含响应式数据的引用对象(reference对象,简称ref对象)

    • JS中操作数据:xxx.value

    • 模板中读取数据: 不需要.value,直接:

      xxx






  • 备注:

    • 接收的数据是:基本类型、对象类型。

    • vue2 里面的ref 是一个属性,类似id,2的到3中也可以使用

    • 基本类型的数据:响应式依然是靠Object.defineProperty()的get与set完成的。

    • 对象类型的数据:内部 “ 求助 ” 了Vue3.0中的一个新函数—— reative函数。

    • ref 处理基本数据类型是get,set(数据劫持);对对象类型数据用的是proxy

2.3 reactive函数

  • 作用: 定义一个对象类型的响应式数据(基本类型不要,要用ref函数)

  • 语法:const 代理对象= reactive(源对象)接收一个对象(或数组),返回一个代理对象(Proxy的实例对象,简称proxy对象)

  • reactive定义的响应式数据是“深层次的”。

  • 内部基于 ES6 的 Proxy 实现,通过代理对象操作源对象内部数据进行操作。






2.4 Vue3中的响应式原理

2.4.1 vue2的响应式

修改this.$set(this.xxx,'xx','xx'),删除vue.delete(),更新this.xxx.splice()

  • 实现原理:

    • 对象类型:通过Object.defineProperty()对属性的读取、修改进行拦截(数据劫持)。

    • 数组类型:通过重写更新数组的一系列方法来实现拦截。(对数组的变更方法进行了包裹)。

      Object.defineProperty(data, 'count', {
          get () {}, 
          set () {}
      })
  • 存在问题:

    • 新增属性、删除属性, 界面不会更新。

    • 直接通过下标修改数组, 界面不会自动更新。

2.4.2  Vue3.0的响应式

proxy handler:对数据增删改查是改变

  • 实现原理:

    • 通过Proxy(代理): 拦截对象中任意属性的变化, 包括:属性值的读写、属性的添加、属性的删除等。

    • 通过Reflect(反射): 对源对象的属性进行操作。

    • MDN文档中描述的Proxy与Reflect:

new Proxy(data, {
	// 拦截读取属性值
    get (target, prop) {
    	return Reflect.get(target, prop)
    },
    // 拦截设置属性值或添加新属性
    set (target, prop, value) {
    	return Reflect.set(target, prop, value)
    },
    // 拦截删除属性
    deleteProperty (target, prop) {
    	return Reflect.deleteProperty(target, prop)
    }
})

proxy.name = 'tom'   

 2.5 reactive对比ref

  • 从定义数据角度对比:

    • ref用来定义:基本类型数据

    • reactive用来定义:对象(或数组)类型数据

    • 备注:ref也可以用来定义对象(或数组)类型数据, 它内部会自动通过reactive转为代理对象

  • 从原理角度对比:

    • ref通过Object.defineProperty()getset来实现响应式(数据劫持)。

    • reactive通过使用Proxy来实现响应式(数据劫持), 并通过Reflect操作源对象内部的数据。

  • 从使用角度对比:

    • ref定义的数据:操作数据需要.value,读取数据时模板中直接读取不需要.value

    • reactive定义的数据:操作数据与读取数据:均不需要.value

2.6 setup的两个注意点

  • vue2里面父传子,子需要用props接收,如果不用就用$atrrs接收

  • 插槽:子组件挖坑里面有$slots,父组件填充

  • setup执行的时机

    • 在beforeCreate之前执行一次,this是undefined。

  • setup的参数

    • props:值为对象,包含:组件外部传递过来,且组件内部声明接收了的属性。

    • context:上下文对象

      • attrs: 值为对象,包含:组件外部传递过来,但没有在props配置中声明的属性, 相当于 this.$attrs

      • slots: 收到的插槽内容, 相当于 this.$slots

      • emit: 分发自定义事件的函数, 相当于 this.$emit。绑定自定义事件需要写

2.7 computed函数

与Vue2.x中computed配置功能一致

import {computed} from 'vue'

setup(){
    ...
	//计算属性——简写
    let fullName = computed(()=>{
        return person.firstName + '-' + person.lastName
    })
    //计算属性——完整
    let fullName = computed({
        get(){
            return person.firstName + '-' + person.lastName
        },
        set(value){
            const nameArr = value.split('-')
            person.firstName = nameArr[0]
            person.lastName = nameArr[1]
        }
    })
}

 2.8 watch函数

  • 与Vue2.x中watch配置功能一致

  • 两个小“坑”:

    • 监视reactive定义的响应式数据时:oldValue无法正确获取、强制开启了深度监视(deep配置失效)。

    • 监视reactive定义的响应式数据中某个属性时:deep配置有效。

//情况一:监视ref定义的响应式数据
watch(sum,(newValue,oldValue)=>{
	console.log('sum变化了',newValue,oldValue)
},{immediate:true})

//情况二:监视多个ref定义的响应式数据
watch([sum,msg],(newValue,oldValue)=>{
	console.log('sum或msg变化了',newValue,oldValue)
}) 

/* 情况三:监视reactive定义的响应式数据
			若watch监视的是reactive定义的响应式数据,则无法正确获得oldValue!!
			若watch监视的是reactive定义的响应式数据,则强制开启了深度监视 
*/
watch(person,(newValue,oldValue)=>{
	console.log('person变化了',newValue,oldValue)
},{immediate:true,deep:false}) //此处的deep配置不再奏效

//情况四:监视reactive定义的响应式数据中的某个属性
watch(()=>person.job,(newValue,oldValue)=>{
	console.log('person的job变化了',newValue,oldValue)
},{immediate:true,deep:true}) 

//情况五:监视reactive定义的响应式数据中的某些属性
watch([()=>person.job,()=>person.name],(newValue,oldValue)=>{
	console.log('person的job变化了',newValue,oldValue)
},{immediate:true,deep:true})

//特殊情况
watch(()=>person.job,(newValue,oldValue)=>{
    console.log('person的job变化了',newValue,oldValue)
},{deep:true}) //此处由于监视的是reactive素定义的对象中的某个属性,所以deep配置有效

watch 时的vue问题

vatch监测的某个属性,使用基本数据类型的监视不用写xx.value,直接写xxx,ref定义的数据

对象在属性值中存的是地址,reactive生成的value,对象监视的是reactive定义的数据

2.9 watchEffect函数

  • watch的套路是:既要指明监视的属性,也要指明监视的回调。

  • watchEffect的套路是:不用指明监视哪个属性,监视的回调中用到哪个属性,那就监视哪个属性。

  • watchEffect有点像computed:

    • 但computed注重的计算出来的值(回调函数的返回值),所以必须要写返回值。)

    • 而watchEffect更注重的是过程(回调函数的函数体),所以不用写返回值。

//watchEffect所指定的回调中用到的数据只要发生变化,则直接重新执行回调。
watchEffect(()=>{
    const x1 = sum.value
    const x2 = person.age
    console.log('watchEffect配置的回调执行了')
})

 2.10 vue 3 生命周期

Vue3.0也提供了 Composition API 形式的生命周期钩子,与Vue2.x中钩子对应关系如下:

  • Vue3.0中可以继续使用Vue2.x中的生命周期钩子,但有有两个被更名:
  • beforeDestroy改名为 beforeUnmount
  • destroyed改名为 unmounted
  • beforeCreate===>setup()
  • created=======>setup()
  • beforeMount ===>onBeforeMount
  • mounted=======>onMounted
  • beforeUpdate===>onBeforeUpdate
  • updated ======>onUpdated
  • beforeUnmount ==>onBeforeUnmount
  • unmounted =====>onUnmounted

 2.11 自定义hook函数

  • 什么是hook?—— 本质是一个函数,把setup函数中使用的Composition API进行了封装。

  • 类似于vue2.x中的mixin。hook函数名一般为useXXX

  • 自定义hook的优势: 复用代码, 让setup中的逻辑更清楚易懂。

2.12 toRef 和 toRefs

  • 作用:创建一个 ref 对象,其value值指向另一个对象中的某个属性。

  • 语法:const name = toRef(person,'name')

  • 应用: 要将响应式对象中的某个属性单独提供给外部使用时。

  • toRefstoRef功能一致,但可以批量创建属性的全部 ref 对象,语法:toRefs(person)

  • ref 本质是(深)拷贝,与原数据无关联,toref的本质是引用(指针),与原始数据相关





你可能感兴趣的:(Vue,vue.js,javascript,typescript)