文章目录
- 回顾Vue2响应式
- Vue3实现响应式
- Ref
- Reactive函数
- 对比reactive和ref
- Vue3组件中的模板结构可以没有根标签div
- 组合式API
- Setup注意点
- 计算属性与监听computed
- Watch
- WatchEffect
- 父组件传值给子组件
- 父组件调用子组件方法
- 子组件调父组件方法
- Vue3生命周期
- 自定义hook函数
- toRef
- 其他组合API
- shallowReactive
- shallowRef
- readonly和shallowOnly
- ToRaw和markRow
- customRef
- Provide和inject
- 响应式数据的判断
- Composition API的优势
- 新的组件
- Fragment
- Teleport
- Suspense
- Vue3其他变化
回顾Vue2响应式
vue2响应式存在的问题:
对象新增属性,删除属性,不会实时响应到界面中。
数组直接通过下标进行修改,不会实时响应到界面中。
vue2中将响应式的数据放到data中,换句话中,只有data中的数据才是响应的。
<script>
export default {
data() {
return {
person: {
name: '张三',
age: 18,
hobby: ['学习', '吃饭']
}
}
},
methods: {
editAddSex() {
// 修改对象中属性用this.$set或者Vue.$set
this.$set(this.person, 'sex', '女')
},
deleteName() {
// 删除对象中属性用this.$delete或者Vue.$delete
this.$delete(this.person, 'name')
},
updateHobby() {
// 修改数组用this.$set或者splice
this.person.hobby.splice(0,1,'逛街')
}
}
}
</script>
Vue2使用object.defineProperty来劫持数据是否发生改变,如下:
<script>
let person = {
name:'风冷螺杆',
type:'三联供'
}
let p={}
Object.defineProperty(p,'name',{
configurable:true,
get(){
console.1og('有人读取了name')
return person.name
}
set(value){
console.1og('有人修改了name')
person.name = value
}
Object.defineProperty(p,'age',{
configurable:true,
get(){
console.log('有人读取了age')
return person.age
}
set(value){
console.1og('有人修改了age')
person.age = value
}
</script>
能监测到获取和修改属性:
新增的属性没有get和set:
删除name属性:
综上,获取和修改一个属性可以捕获到,但是新增和删除属性是捕获不到的,所以使用
s
e
t
和
set和
set和delete方法才能响应。
Vue3实现响应式
- 如下代码,p = new Proxy()后,p是一个proxy对象,是person的一个代理对象,对p的修改会映射到person身上:
let person = {
name: '风冷螺杆',
type: '三联供'
}
let p= new Proxy(person, {})
上面代码只是说P是person的代理对象,p改变,person跟着改变,但是并没有做到响应式,也就是变化并没有捕获到。
2) Vue3捕获响应:
结果如下:对对象的增删改查全部捕获到,并且映射到person中。
其中set方法,既捕获新增又捕获修改。
3) 对上面代码进行优化,vue3中是使用Reflect来对数据进行修改。
综上,实现原理: 通过proxy(代理):拦截对象中任意属性的变化,包括:属性值的读写,属性的添加,属性的删除等。
通过Reflect(反射):对被代理对象(原对象)的属性进行操作。
Ref
RefImpl:引用对象,如果想让一个普通变量变成响应式的,就需要把这个变量丢给ref。
<template>
<h1>设备名称:{{ name }}</h1>
<h1>寿命:{{ age }}</h1>
<button @click="sayHello">点击</button>
</template>
<script>
import { ref } from 'vue'
export default {
name: 'App',
setup() {
// 数据
let name = ref('风冷模块')
let age = ref(18)
// 方法
function sayHello() {
// name.value = '变频'
// age.value = 20
console.log(name, age)
}
// 返回响应式数据和方法给模板使用
return {
name,
age,
sayHello
}
}
}
</script>
修改的时候需要使用name.value进行修改。使用的时候直接使用name字段就行。
补充:const obj= reactive({name:’li’})定义的对象是可以修改对象里面的属性,如:obj.name = ‘ni’
但是不可以修改整个对象,比如:obj = {name:’ni’}
Ref作用:定义一个响应式的数据。
Ref语法:
1)创建一个包含响应式数据的引用对象
// 引用对象,简称ref对象
let name = ref('风冷模块');
let age = ref(18);
let job = ref({
type = '恒温恒湿',
value = 20
})
2)JS中操作数据,使用XXX.value。
3)模板中使用,不需要value,直接{{XXX}}。
Ref接收的数据类型:基本类型,对象类型。
const a = ref(1); Const a = ref(‘1’); const a = ref(true); const a = ref({}); const a = ref([])
基本数据类型:响应式依靠的是object.defineProperty()的get和set。
对象类型:内部求助了vue3的一个新函数——reactive函数。
Reactive函数
作用:定义一个对象类型的响应式数据(基本类型不要用它,要用ref函数)。
使用ref定义的响应式数据,在js中修改的时候必须加上.value;使用reactive定义的响应式数据,不用.value就可以进行修改,无论对象层级有多深,数组也可以直接用索引来修改。
语法:const 代理对象 = reactive(原对象),接受一个对象或数组,返回一个代理对象(proxy的实例对象,简称proxy对象)。
Reactive定义的响应式数据是’深层次’的。
内部基于ES6的proxy实现,通过代理对象操作源对象内部数据进行操作。
对比reactive和ref
- 从定义数据角度对比
Ref用来定义:基本数据类型。
Reactive用来定义:对象(或数组)类型数据。
备注:ref也可以定义对象(或数组)类型数据,它内部会自动通过reactive转为代理对象。 - 从原理角度对比
Ref通过object.defineProperty()的get和set来实现响应(数据劫持)。
Reactive通过proxy来实现响应式(数据劫持),通过Reflect来操作源数据内部。 - 从使用角度对比
Ref定义的数据,操作数据需要使用.value,模板中使用数据不需要使用.value。
Reactive定义的数据,操作数据与读取数据,均不需要使用.value。
Vue3组件中的模板结构可以没有根标签div
<template>
<img alt="Vue logo" src="./assets/logo.png">
<HelloWorld msg="Welcome to Your Vue.js App"/>
</template>
<script>
import HelloWorld from './components/HelloWorld.vue'
export default {
name: 'App',
components: {
HelloWorld
}
}
</script>
组合式API
- Setup
Vue3中的一个新的配置项,是一个函数;
组件中所用到的数据、方法等等,均要配置在setup中;
Setup函数的两种返回值:
1)若返回一个对象,则对象的属性、方法,在模板中均可以直接使用。(常用)
<template>
<h1>设备名称:{{ name }}</h1>
<h1>寿命:{{ age }}</h1>
<button @click="sayHello">点击</button>
</template>
<script>
export default {
name: 'YourComponentName', // 需要替换为你的组件名
setup() {
// 数据
let name = '风冷模块';
let age = 18;
// 方法
function sayHello() {
alert(`设备名称:${name}`);
}
return {
name,
age,
sayHello
};
}
}
</script>
2)若返回一个渲染函数,可以自定义渲染内容。
<template>
<h1>设备名称:{{ name }}</h1>
<h1>寿命:{{ age }}</h1>
<button @cLick="sayHello">点击</button>
</template>
<script>
import { h } from 'vue' // h函数需要手动引入
export default {
name: 'App',
setup() {
return () => h('h1', '你好啊') // 渲染函数
}
}
</script>
注意点:
1、尽量不要与vue2.x配置使用
*Vue2配置(data,methods,computed…)中可以访问到vue3的setup中的属性、方法。
*vue3的setup中的配置不能访问到Vue2的配置(data,methods,computed…)。
*如果有重名,setup优先。
2、setup不能是一个async函数,因为如果是async函数,返回值不再是return的对象,而是promise,模板中将看不到return中的属性。
Setup注意点
- Setup执行时机:beforeCreate之前执行一次,这个时候的this是undefined。
Setup一共有两个参数,第一个是父组件传过来的值:
第二个参数是:context
Context.attrs:值为对象,包含:组件外部传递过来,但是没有在props配置中声明的属性,相当于this.
a
t
t
r
s
。
C
o
n
t
e
x
t
.
e
m
i
t
:分发自定义事件的函数,相当于
t
h
i
s
.
attrs。 Context.emit:分发自定义事件的函数,相当于this.
attrs。Context.emit:分发自定义事件的函数,相当于this.emit。(下面举例)
Context.slots:收到的插槽内容,相当于this.$slots。(下面举例)
Context.emit使用:
Context.slots使用:
计算属性与监听computed
在 Vue 3 的 Composition API 中,computed 函数用于创建计算属性。计算属性是基于它们的响应式依赖进行缓存的,只有在它的相关依赖发生改变时才会重新求值。这使得计算属性在处理复杂逻辑时非常高效,因为它们可以避免不必要的重复计算。
以下是如何在 Vue 3 的 Composition API 中使用 computed 的示例:
import { computed, reactive } from 'vue'
export default {
name:'App',
setup() {
let person = reactive(
{
surname: '张',
name: '三'
}
)
//计算属性--简写(没有考虑计算属性被修改的情况)
person.fullname = computed(() => {
return person.surname +'-'+person.name
})
}
}
注意,computed 函数接收一个 getter 函数作为参数,该 getter函数返回一个基于响应式依赖的值。当依赖项变化时,computed 会重新计算 getter 函数的返回值,并更新计算属性的值。
你还可以使用 computed 的第二个参数(可选)来提供一个 setter 函数,以便在需要时手动设置计算属性的值。但是,由于计算属性通常是基于其他响应式依赖进行计算的,所以 setter 函数的使用并不常见:
import { computed, reactive } from 'vue'
export default {
name:'App',
setup() {
let person = reactive(
{
surname: '张',
name: '三'
}
)
//计算属性--简写(没有考虑计算属性被修改的情况)
person.fullname = computed(() => {
get() {
return person.surname+'-'+person.name
},
set(value) {
const nameArr=value.split('-')
person.surname = nameArr[o]
person.name = nameArr[1]
}
})
}
}
Watch
Watch在vue3中是一个组合API,可以多次调用,它有三个参数:
Params1:被监听的变量,可以是一个数组,存放多个变量。
Params2:回调函数,监听的数据有变化时调用,回调函数中有两个参数,newVal和oldVal。
Params3:是一个对象,存放watch的配置{immediate:true,deep:true}。
- Watch监听ref变量
- Watch监听reactive变量
Vue3bug:
1)watch监听reactive变量时,oldValue值获取不到。
2)强制开启了深度监听,deep配置无效。
在vue3中,不管嵌套的层级有多深,只要是用reactive定义的,都可以监听到。但是获取不到正确的oldVal。
情况三:监听reactive定义的嵌套多级的对象,不管嵌套层级有多深,都是深度监听
情况四,监听reactive定义的响应式对象中的某一属性。使用回调函数的方式。并且可以获取到oldValue值。
情况五:监视reactive定义的响应式对象的某些属性。可以成功获取oldVal。
特殊情况:监听salary,要使用deep。拿不到oldValue。
总结:
// 监听ref定义的变量:不用使用deep,可以成功获取到oldValue
// 情况一:监视ref所定义的一个响应式数据
watch(sum, (newValue, oldValue) => {
console.log('sum变了', newValue, oldValue);
}, { immediate: true });
// 情况二:监视ref所定义的多个响应式数据
watch([sum, msg], (newValue, oldValue) => {
console.log('sum或msg变了', newValue, oldValue);
}, { immediate: true });
// 情况三:监视reactive所定义的一个响应式数据的全部属性
// 1.注意:此处无法正确的获取oldValue
// 2.注意:强制开启了深度监视(deep配置无效)
// 直接监听reactive定义的变量,无法获取oldValue;使用deep:false无效
watch(person, (newValue, oldValue) => {
console.log('person变化了', newValue, oldValue);
}, { deep: false }) // 此处的deep配置无效
// 情况四:监视reactive所定义的一个响应式数据中的某个属性
// 监听reactive定义的对象中的某些非对象属性:不用使用deep;能获取到oldValue
watch(() => person.name, (newValue, oldValue) => {
console.log('person的name变化了', newValue, oldValue);
})
// 情况五:监视reactive所定义的一个响应式数据中的某些属性
watch([() => person.name, () => person.age], (newValue, oldValue) => {
console.log('person的name或age变化了', newValue, oldValue);
})
// 特殊情况
// 监听reactive定义的对象中的对象类型的属性:使用deep:true深度监听;无法获取到oldValue
watch(() => person.job, (newValue, oldValue) => {
console.log('person的job变化了:', newValue, oldValue);
}, { deep: true });
答疑:
1、为什么监听ref定义的基本数据类型不需要使用.value,监听ref定义的引用类型的数据需要使用.value。
监听ref定义的基本数据类型:监听的是RefImpl对象。
监听ref定义的引用类型数据,监听的是RefImpl对象,但是RefImpl中的value值是Proxy对象,需要深度监听,所以需要使用{deep:true},或者监听person.value。
WatchEffect
Watch:既要指明监视的属性,又要指明监视的回调。
WatchEffect:不用指明监视的属性,监视的回调中用到了哪个属性,就监视哪个,而且深度监听。
WatchEffect有点像computed:
1)但computed注重的是计算出来的值(回调函数的返回值),所以必须要写返回值。
2)而WatchEffect注重的是过程(即函数体),所以不用写返回值。
如下例子,person.salary.s1.sa发生变化了就会被监听到。
父组件传值给子组件
// 父组件
在父组件中,使用 v-bind 的缩写 : 来动态地绑定 currentSelected 到子组件的 prop。这样,当 currentSelected 的值改变时,子组件的 prop 也会自动更新。
<template>
<div class="filter">
<selectDate :currentSelected="currentSelected"/>
</div>
</template>
<script lang="ts" setup>
import selectDate from "@/views/dataStatistic/children/selectDate.vue";
</script>
// 子组件
在子组件中,需要使用 props 选项来声明你期望从父组件接收哪些属性。
import { defineProps, withDefaults } from "vue";
interface Props {
currentSelected: string;
}
let props = withDefaults(defineProps<Props>(), {
currentSelected: '',
});
// 使用
console.log(props.currentSelected)
使用对象形式来定义 prop,这样可以提供验证功能。例如,你可以指定 prop 的类型、默认值或是否必须。在上面的例子中,我们指定了 currentSelected 必须是一个字符串。
在子组件中,可以使用 watch 选项来监听 prop 的变化,并在变化时执行某些操作。但是请注意,尽量避免在子组件中直接修改prop,因为这可能会导致数据流变得混乱。如果需要在子组件中修改数据,应该使用 data 属性或 computed属性来维护一个内部状态,并根据 prop 的值来更新这个状态。
父组件调用子组件方法
使用 $refs
在 Vue 3 中,你可以使用 ref 属性在模板中注册一个子组件的引用。然后,在父组件中,你可以通过 this.$refs 访问这个子组件实例,并调用其方法。
父组件
<template>
<section class="dataList">
<provincesDataList ref="provincesDataRef"/>
</section>
</template>
<script lang="ts" setup>
import provincesDataList from "../provincesDataList.vue";
const provincesDataRef = ref()
// 调用子组件getDataListFn方法
provincesDataRef.value.getDataListFn()
</script>
子组件:
<script lang="ts" setup>
import { ref, defineExpose } from "vue";
const getDataListFn = async () => {
// 方法体
}
defineExpose({
getDataListFn
});
</script>
子组件调父组件方法
在 Vue 3 中,子组件通常不会直接调用父组件的方法,因为这同样违反了组件之间的单向数据流原则。然而,你可以通过$emit方式实现子组件通知父组件并执行某些操作:
父组件
子组件可以通过 $emit 触发一个自定义事件,并传递参数给父组件。父组件在模板中监听这个事件,并在事件触发时执行相应的方法。
<template>
<div>
<selectDate @get-data="getDataListFn"/>
</div>
</template>
<script lang="ts" setup>
import selectDate from "@/views/dataStatistic/children/selectDate.vue";
const getDataListFn = async (data: any) => {
// 从子组件传过来的数据
console.log(data) // {name: 'ying'}
}
</script>
子组件
import { defineEmits } from "vue";
emit('getData', {name: 'ying'}});
const emit = defineEmits<{
(e: 'getData', data: any): void;
}>();
子组件的getData和父组件的@get-data对应即可
Vue3生命周期
第一种使用方式——配置项的方式
注意:
1)使用配置项这种方式的生命周期,setup在所有生命周期之前执行。
2)最后两个生命周期在vue2中是beforeDestroy和destroyed,vue3中改成了beforeUnmount和unmounted。
第二种使用方式——组合式API
注意:
1)使用组合API生命周期钩子,onBeforeCreate和onCreated两个钩子没有API,就相当于setup。
自定义hook函数
Vue3的hook函数,相当于vue2的 mixin,不同在于hooks是函数。
Vue3的hook函数,可以帮助我们提高代码的复用性, 让我们能在不同的组件中都利用 hooks 函数。
例子:将鼠标点击页面就获取坐标点的方法拿出独立,便于其他组件复用。
toRef
作用:创建一个ref对象,其value值指向另一个对象中的某个属性。
语法:const name = toRef(device,’name’).
应用:要将响应式对象中的某个属性单独提供给外部使用时。
扩展:toRefs和toRef功能一样,但是可以创建多个ref对象,语法:toRefs(device)。
其他组合API
shallowReactive
作用:浅层次的响应式,只考虑第一层的响应式。
如下举例:name和age会响应,但是position.po.p不是响应式的。
shallowRef
只处理基本数据类型的响应式,不进行对象的响应式处理。
如果使用ref,处理对象时,内部借用的是reactive方法,使用ES6的proxy,shallowRef没有做这个操作,所以不能对对象进行响应式处理。
何时使用?
1)如果有一个对象,结构比较深,但是变化时只是外层属性发生变换,使用shallowReactive。
2)如果有一个对象,后续功能不会修改该对象里面的属性,而是生成新的对象来替换,使用shallowRef。
readonly和shallowOnly
readonly:是一个函数,接收一个响应式的数据,相当于将这个响应式的数据保护起来,不能进行修改。
shallowOnly:让一个响应式的数据变为只读的,不可修改(浅只读),浅只读就是对于嵌套层级的对象来说,只能对第一级进行只读操作,其他级依然是响应式。
应用场景:不希望数据被修改时。
ToRaw和markRow
markRow应用几率高。
ToRaw:把reactive定义的响应式数据转成原始数据(没有响应式)。
使用场景:用于读取响应式对象对应的普通对象,对这个普通对象的所有操作,不会引起页面更新。
注:ref定义的无效。
markRow:标记一个对象,使其永远不会再成为响应式对象。
应用场景:
1、有些值不应被设置为响应式,例如复杂的第三方类库;
2、当渲染具有不可变数据源的大列表时,跳过响应式转换可以提高性能。(例如一个层级很深很深的数据,vue3会一层一层的做响应式,如果这个数据只是展示使用,不需要对其进行修改,可以使用markRaw标记为普通对象,从而提高性能。)
小插曲:如果动态给一个响应式的对象中添加属性,要么一开始定义这个对象的时候就先将这个属性定义出来;如果使用toRef(),就需要先将这个大对象抛出去,否则第一次进去页面报错。
给reactive定义的响应式数据身上追加任何属性,都是响应式的属性。
使用markRaw标记后,car不再是响应式的。
customRef
作用:创建一个自定义的ref,并对其依赖项跟踪和更新触发进行显式控制。
注:customRef函数调用要传入一个函数,函数体里面要返回一个对象。
举例:实现防抖效果(在输入框中输入的东西,过500ms再在下面显示出来)
Provide和inject
作用:实现祖和后代组件(跨级)间通信。
套路:父组件有一个provide选项来提供数据,后代任何一级的组件(包括子组件,即第一级组件,但是没必要,子组件使用props就好。)有一个inject选项来开始使用这些数据。
响应式数据的判断
isRef:检查一个值是否为一个ref对象。
isReactive:检查一个对象是否是由reactive创建的响应式代理。
isReadonly:检查一个对象是否是由readonly创建的只读代理。
isProxy:检查一个对象是否是由reactive或者readonly方法创建的代理。
Composition API的优势
使用传统的opsitionApi,新增或者修改一个需求,就需要分别在data,methods,computed中修改。使用compositionAPI我们可以更优雅的组织我们的代码,函数,让相关功能的代码更加有序的组织到一起。
新的组件
Fragment
在vue2中,组件要有一个根标签;
在vue3中,组件可以没有根标签,内部会将多个标签包裹在一个虚拟的Fragment元素中。好处:减少标签层级,减少内部占用。
Teleport
Teleport是一种能将我们的组件html结构传送到指定位置的技术。
举例:父组件中引入子组件,子组件中引入孙组件,孙组件中引入dialog组件:
正常来讲,html结构是:弹窗在孙组件中,不能移动到body体内。
使用teleport包裹后,可以将包裹的内容移动到任意位置
将弹窗相关的html结构移动到了body的位置。
Suspense
等待异步组件时渲染一些额外内容,让用户有更好的体验。
使用步骤:
举例:将子组件放在suspense中包裹起来,再异步引入。
显示效果:子组件显示显示‘稍等,加载中…’,再显示child组件中真正的内容。
补充内容:
之前说过,setup不能是一个async函数,因为返回值不再是return的对象,而是promise,模板看不到return中的属性。
但其实,setup可以是async函数,也可以返回一个promise对象,但需要suspense和异步引入组件的配合。
让组件等一等再加载出来的两种方法:
1、使用了异步组件,异步引入,suspense,网速慢;
2、使用promise。
Vue3其他变化
例如如下两种写法,都是依赖按键code。
移除:@keyup.13 = “fn” 的写法;
移除:Vue.config.keyCodes.huiche = 13 //定义了一个别名按键