Vue3.0 | vue3的新特性
Vue3的变化
官网地址: https://v3.cn.vuejs.org/guide/migration/introduction.html
一、对比vue2的变化
1.优点
- vue3支持vue2的大多数特性,实现对vue2的兼容
- vue3对比vue2具有明显的性能提升
- 打包大小减少41%
- 初次渲染快55%,更新快133%
- 内存使用减少54%
- 更好的支持TypeScript
- 使用Proxy代替defineProperty实现响应式数据
2.性能提升的原因
- 静态标记
- vue2从根节点开始对虚拟dom进行全量对比(每个节点不论写死的还是动态的都会一层一层比较)
- vue3新增了静态标记 与上次虚拟dom对比的时候,只对比带有 patchFlags 的节点。跳过一些静态节点对比(下图编译结果中-1跟1就属于静态标记)
patchFlags配置文件地址 https://github.com/vuejs/core/blob/main/packages/shared/src/patchFlags.ts
helloWorld{{msg}} - hoistStatic 静态提升
- vue2里每当触发更新的时候,不管元素是否参与更新,每次都会重新创建
- vue3为了避免每次渲染的时候都要重新创建这些对象,会把不参与更新的元素保存起来,只创建一次,每次复用。比如上面_hoisted_1,_hoisted_2被提升到渲染函数render之外,
- cacheHandlers 事件缓存
- vue2里绑定事件都要重新生成新的function去更新
- vue3会自动生成一个内联函数,同时生成一个静态节点。onclick时会读取缓存,如果缓存没有的话,就把传入的事件存到缓存里
点击
3.响应式数据的变化
- vue2的响应式(实现地址:https://github.com/jiaomengyuan)
- 核心
- Object.defineProperty 来劫持各个属性的setter、getter,在数据变动时发布消息给订阅者,触发相应的监听回调
- 对象:递归调用defineProperty对对象已有属性值的读取和修改进行拦截
- 数组:重写数组更新数组一系列更新元素的方法来实现元素修改的劫持
- 问题
- 不能监听到对象属性的添加和删除,需要Vue.set()来添加和删除。
- 不能通过下标替换元素或更新length
- 核心
- vue3的响应式(实现地址:https://github.com/jiaomengyuan)
- 核心
- 通过Proxy(代理): 拦截对data任意属性的任意(13种)操作, 包括属性值的读写, 属性的添加, 属性的删除等
- 通过 Reflect(反射): 动态对被代理对象的相应属性进行特定的操作
- 核心
二、 新增特性
1.Composition (组合) API
-
Option API
:vue2创建组件时,会把数据放到data,计算属性放到computed,事件处理放到methods,监听改变放到watch;共同处理页面逻辑- 组件功能越来越多,逻辑功能点分散,不易阅读(新增或修改一个需求,就需要分别在data,methods等...进行修改,功能多时,滚动条来回滚动 )
- 可以通过Mixins重用逻辑代码,但是数据来源模糊还会有Mixins命名冲突的问题
-
Composition API
:将零散的data,methods代码重新组合,一个功能的代码放一块儿,并且可以单出拆分出函数- 兼容Option API,还可继续使用
- 利于代码重用,没有对this的使用,减少了this指向不明的情况
- 几乎是函数,编辑器可以帮我们进行类型检查和建议
2. setup
- setup函数是一个新的option,在初始化时执行一次,可以理解为使用Composition API 的入口点。
- 这个函数的返回一个对象,对象里的属性和方法,可以直接在模版中使用
{{msg}}
注意:
- 在beforeCreate之前创建,因此,这个函数中没有this。因此不能访问data,methods等。但methods中可以访问setup提供的属性和方法
- return中返回的属性跟data合并,返回的方法跟methods里的方法合并;如有重名,setup优先
- setup不能是一个async函数,使用async后返回值不是return的对象,而是promise。非要使用,需要使用
包裹组件 - setup接收两个参数
setup(props, context)
||setup(props, {attrs, slots, emit})
不能解构props,否则会失去响应式
1) ref
- 定义一个基本数据类型的响应式引用
{{num}}
注意:
- 在setup中使用ref定义的响应式引用需要
.value
(内部通过给value属性添加getter、setter实现对数据的劫持),在模版中不需要(解析模板时会自动添加.value) - ref常用来处理基本数据类型,如果用ref定义引用数据类型, 内部会自动将对象,数组转换为reactive的代理对象
2) reactive
- 定义多个数据的响应式引用
{{ obj.name }}
{{ obj.age }}
3) toRef(obj,key) 、 toRefs(obj)
- reactive定义的响应式数据,解构后进行使用,数据就不是响应式的,使用
toRef
或toRefs
将解决这个问题 -
toRef
跟toRefs
都是复制reactive里的属性然后转成ref。因为是浅拷贝,所以修改复制出来的值时原来reactive里的数据也会跟着更新 -
toRef
是复制reactive里的单个key转成ref,toRefs
复制reactive里的所有key转成ref - 也可以用来将props接收的数据进行解构响应式
{{ obj.name }}
{{ obj.age }}
{{ name }}
{{ age }}
4) shallowRef、triggerRef 与 shallowReactive
-
ref
和reactive
是把数据变为响应式,无论层级多深,始终都会对数据进行深度监听。shallowRef
和shallowReactive
可以解决这个 -
shallowRef
只处理了.value的响应式(只监听.value的变化)。因此如果需要修改应该xxx.value={}
或者使用shallowRef
将数据强制更新到页面 - 注:
shallowReactive
的数据使用shallowRef
强制更新无效
{{ obj.name }}
{{ obj.age }}
{{ obj.c.d }}
-
shallowReactive
只处理对象最外层属性的响应式(浅响应式),默认情况下只能监听第一次的数据。如要修改则需要先改第一层的数据,然后再去更改其他层的数据
{{ obj.name }}
{{ obj.age }}
{{ obj.c.d }}
5) readonly 与 shallowReadonly
-
readonly
是让一个响应式数据只读,深层只读
{{ obj.name }}
{{ obj.age }}
{{ obj.c.d }}
-
shallowReadonly
是让一个响应式数据只读,浅层只读(响应式对象的最外层只读,但再深一层的属性可以被修改)
{{ obj.name }}
{{ obj.age }}
{{ obj.c.d }}
6) toRaw 、 markRaw
-
toRaw
将一个响应式对象(由 reactive定义的响应式)转换为普通对象
{{ obj.name }}
{{ obj.age }}
{{ obj.c.d }}
-
markRaw
标记一个对象,使其不能成为一个响应式对象
{{ obj.name }}
{{ obj.age }}
{{ obj.c.d }}
7) isRef 、unref 、 isReactive 、isProxy 、isReadonly
-
isRef
检查值是否为一个 ref 对象。unref
为val = isRef(val) ? val.value : val
的语法糖
const obj = ref({ name: "张三", age: 25, c: { d: 2 } });
console.log(isRef(obj))
console.log(unref(obj))
-
isReactive
检查值是否为一个 reactive 对象
const obj = reactive({ name: "张三", age: 25, c: { d: 2 } });
console.log(isReactive(obj))
-
isProxy
检查对象是否是由 reactive 或 readonly 创建的 proxy
const obj = reactive({ name: "张三", age: 25, c: { d: 2 } });
console.log(isProxy(obj))
const obj = readonly({ name: "张三", age: 25, c: { d: 2 } });
console.log(isProxy(obj))
-
isReadonly
检查对象是否是由 readonly 创建的只读代理
const obj = readonly({ name: "张三", age: 25, c: { d: 2 } });
console.log(isReadonly(obj))
8) customRef
- 创建一个自定义的ref,接受一个函数作为参数,这个函数接受两个参数
track
(通知vue需要追踪后续内容的变化)trigger
(通知vue重新解析模版)
{{inpValue}}
9) computed、watch、watchEffect
- 在setup中也有单独的
computed
和watch
用法基本相同 -
watchEffect
是监视所有回调中使用的数据,因为每次初始化的时候都会执行一次回掉自动获取依赖,并不用手动传需要监听的对象。注:无法获取到oldValue
{{ obj.name }}
{{ obj.age }}
注意:
当使用 watch 选项侦听数组时,只有在数组被替换时才会触发回调。换句话说,在数组被改变时侦听回调将不再被触发。要想在数组被改变时触发侦听回调,必须指定 deep 选项。
10) 生命周期
- setup中也有新的生命周期
onBeforeMount
->onMounted
->onBeforeUpdate
->onUpdated
->onBeforeUnmount
->onUnmounted
->onErrorCaptured
跟options api混用时onBeforeMount在beforeMount之前,onMounted在mounted之前。。。之后都是 - vue中父子顺序
父beforeCreate
->父created
->父beforeMount
->子beforeCreate
->子created
->子beforeMount
->子mounted
->父mounted
在setup中声明周期也适用
11) provide 、 inject
-
provide
、inject
提供依赖注入 ,实现祖孙级组件通信
父组件
子组件
12) $refs
- 在vue2中使用
this.$refs.XXX
获取,vue3中setup函数没有this,所以也有单独的获取ref的方法
13) 自定义hook函数
- 与mixin类似,抽离公共代码
import { reactive } from "vue";
export default function hookTest() {
const obj = reactive({ name: "张三", age: 25 });
return { obj };
}
{{ obj.name }}
{{ obj.age }}
3. 其他新特性
1) 全局API
- 在vue2中的main.js中有以下代码。如果使用全局api则是 Vue.directive、Vue.component、Vue.config、Vue.mixin、Vue.prototype等,都是挂载在Vue原型上
import Vue from 'vue'
import App from './App.vue'
const app = new Vue(App)
app.$mount()
- vue3提供的是实例api。通过createApp创建vue实例
import { createApp } from 'vue'
import App from './App.vue'
createApp(App).mount('#app')
2) v-if 与 v-for 的优先级对比
- 在vue2中:当v-if和v-for同时使用时,v-for的优先级高于v-if(因此我们通常需要计算属性先对数据进行加工处理,以达到性能优化的目的)
- 在vue3中:当v-if和v-for同时使用时,v-if的优先级高于v-for
3) v-for 中的 Ref 数组
- vue2中,在 v-for 语句中使用ref属性,会生成refs数组插入$refs属性中
{{ item }}
this.$refs将会是个数组
- vue3中,在v-for中使用ref属性,将不会自动在$refs里创建数组,而是将ref绑定到一个函数中,在函数中可以处理ref
{{ item }}
4) v-bind合并行为
- vue2如果一个标签同时定义了动态属性和一个相同的单独的属性,那么这个单独的属性总是会覆盖动态属性。说明单独属性优先级高于动态属性
- vue3将会吧动态属性跟单独属性合并
5) v-model
- vue2中,在组件上使用v-model默认prop与event为
value
和input
。如要修改,通过子组件的model选项中的prop值和event值来指定属性名和事件名。
export default {
model: {
prop: 'title',
event: 'change'
},
props: {
title: {
type: String,
default: 'Default title'
}
},
methods:{
handleClick(val) {
this.$emit('change', val)
}
}
}
- 除了使用上面方法对某一个prop进行 ‘双向绑定’ 还可以通过这种方式
v-bind.sync
- vue3中v-bind 的 .sync 修饰符已移除。
:title.sync
就要替换为v-model:title
this.$emit('update:title', newValue)
- vue3默认prop与event为
modelValue
和update:modelValue
。如要修改,直接通过v-model
后面参数v-model:title
来指定属性名,并且支持绑定多个v-model
。
- 如果需要修改model的名称,我们可以为v-model传递一个参数,作为子组件内model选项的代替
6) $attrs
- vue2中使用
v-bind='$attrs'
进行将父组件不被认为props特性绑定的属性传入子组件(不包含class以及style),配合interitAttrs一起使用,如果为true则将所有attribute添加到子组件的跟标元素上。但如果为false时,因为class以及style不属于$attrs,所以仍会添加到组件的跟元素上 - vue3中$attrs包含所有传递给子组件的attribute,包含class以及style
7) emits选项
- vue2中子组件触发父亲组件的方法
this.$emit(方法名)
- vue3提供了一个类似props的emits选项,emits选项可以配置校验emit事件,为null的时候不校验。校验时会把参数携带过去,当校验不通过,控制台会发出一个警告,但emit事件还会继续执行
- 官方建议我们在组件中所有的emit事件都能在组件的emits选项中声明
点击
7) 事件 API(eventBus)
- 移除
$on
,$off
和$once
实例方法,组件实例不再实现事件触发接口。
8) 函数式组件
- 函数式组件有两个特性1、Stateless无状态:组件自身没有状态 2.Instanceless无实例:组件自身没有实例,也就是没有this
- vue2中,函数式组件通常用为性能优化,它的初始化速度比有状态组件快的多
Vue.component('typeButton',typeButton)
const typeButton = {
functional:true,//标记,无状态无实例
render(h , { props }){
const { type } = props
return {type}
}
}
- vue3中不需要
functional
定义 接收两个参数,props和context。context包含组件的attrs
、slots
和emit
property
import { h } from 'vue'
const DynamicHeading = (props, context) => {
return h(`h${props.level}`, context.attrs, context.slots)
}
DynamicHeading.props = ['level']
export default DynamicHeading
9) 异步组件 defineAsyncComponent
const asyncModal = () => import('./Modal.vue')
//带有选项
const asyncModal = {
component: () => import('./Modal.vue'),
delay: 200,
timeout: 3000,
error: ErrorComponent,
loading: LoadingComponent
}
- vue3中函数式组件被定义为纯函数,因此异步组件需要包裹在
defineAsyncComponent
中
import { defineAsyncComponent } from 'vue'
const asyncModal = defineAsyncComponent(() => import('./Modal.vue'))
//带有选项
const asyncModalWithOptions = defineAsyncComponent({
loader: () => import('./Modal.vue'),
delay: 200,
timeout: 3000,
errorComponent: ErrorComponent,
loadingComponent: LoadingComponent
})
10) data选项
- vue2中,可以通过Object或者function定义data选项
- vue3中,data只接受返回object的function
- 另外于mixin合并操作将被浅层次执行(data和mixin中声明同样的对象,vue2会把对象中的合并,vue3将只取mixin中)
三、 新组件
1. setup语法糖
- setup这个option,暴露变量必须都return出来,模板中才能使用。vue3也提供了相关语法糖,不用export ,也不需要写setup函数
- 需要将 setup attribute 添加到
代码块上.变量跟函数都不需要return,即可在模板中使用
- 引入组件可以直接使用,不需要通过components进行注册
{{ obj.name }}
{{ obj.age }}
2. style
3. Fragment(片断)
4. Teleport(瞬移)
-
用于移动dom到指定元素 - 还可配置
disabled
属性,禁用teleport的功能,不会移动到任何位置,但他仍会出现在正常位置下(里面的所有元素依然保持正常状态)
5. Suspense -- 实验
- 组件渲染被渲染之前,可能存在异步操作,
帮助我们创建一个平滑的用户体验 -
有两个插槽default、fallback。如果default里的标签(任意深度,所有的后代组件)显示不出来,则展示fallback - fallback被触发的机制还有上面说的setup是一个async函数。这样return的则是promise
Loading...
四、 废除属性
1. $children
- 在vue2中,可以使用
this.$children
访问当前实例的子组件。vue3中已移除,推荐使用$refs
2. 过滤器filter
- 在vue2中,可以使用过滤器来处理数据格式。vue3中已移除,推荐使用方法调用或计算属性
3. $listeners
- vue2可以使用
v-bind='$attrs'
访问父组件不被认为的props特性绑定的属性传入子组件,还可以通过在中间组件添加v-on=‘$listeners’
使其子组件可以通过emits触发其父组件的方法 - vue3将移除这一方法,因为现在事件监听器是
$attrs
的一部分(相当于父组件的@init=‘init’。不用在中间组件添加v-on=‘$listeners’
,只需要v-bind='$attrs'
,由$attrs负责传递方法。即可在其子组件访问)
4. inline-template
- 当 inline-template 出现在一个子组件上时,这个组件里面的内容作为模版,而不是将其作为被分发的内容
- vue3将移除这一属性。可以通过默认插槽进行实现
5. propsData
- vue2中使用propsData用于在创建Vue实例的过程中传入prop
- vue3移除了这一属性。如果需要传递,则需要createApp的第二个参数
const Comp = Vue.extend({
props: ['username'],
template: '{{ username }}'
})
new Comp({
propsData: {
username: 'Evan'
}
})
//vue3中需要这样
const app = createApp(
{
props: ['username'],
template: '{{ username }}'
},
{ username: 'Evan' }
)
共有 0 条评论