title: Vue3 总结
date: 2022-11-06 23:57:57
tags:
Vue2 开发见:Vue2 总结(开发)_凡 223 的博客
性能的提升
源码的升级
拥抱 TypeScript
Vue3 可以更好的支持 TypeScript
新的特性
官方文档:https://cli.vuejs.org/zh/guide/creating-a-project.html#vue-create
## 查看@vue/cli版本,确保@vue/cli版本在4.5.0以上
vue --version
## 安装或者升级你的@vue/cli
npm install -g @vue/cli
## 创建
vue create vue_test
## 启动
cd vue_test
npm run serve
官方文档:https://v3.cn.vuejs.org/guide/installation.html#vite
vite官网:https://vitejs.cn
什么是 vite?—— 新一代前端构建工具。
传统构建 与 vite 构建对比图


## 创建工程
npm init vue@latest
或 npm create vite@latest
## 进入工程目录
cd <project-name>
## 安装依赖
npm install
## 运行
npm run dev
// 引入的不再是Vue构造函数,引入的是一个名为 createApp 的工厂函数(不需要 new)
import { createApp } from 'vue'
import App from './App.vue'
// createApp(App).mount('#app')
// 创建应用实例对象——app(类似于之前Vue2中的vm,但app比vm更“轻”)
const app = createApp(App)
//挂载
app.mount('#app')
Vue3 组件中的模板结构可以没有根标签
<template>
<!-- Vue3组件中的模板结构可以没有根标签 -->
<img alt="Vue logo" src="./assets/logo.png">
<HelloWorld msg="Welcome to Your Vue.js App"/>
</template>

<template>
<h1>姓名:{{name}}h1>
<h1>年龄:{{age}}h1>
<button @click="info">个人信息button>
template>
<script>
// import {h} from 'vue'
export default {
name: 'App',
setup() {
// 数据
let name = "张三"
let age = 18
// 方法
function info() {
alert(`我叫${name}, 年龄${age}岁`)
}
// 返回一个对象(常用)
return {
name,
age,
info
}
//返回一个函数(渲染函数),需要导入 h
// return ()=> h('h1','渲染')
}
}
script>

1、尽量不要与 Vue2.x 配置混用
2、setup 不能是一个 async 函数,因为 async 的返回值不再是 return 的对象,而是被 promise 包起来的,模板看不到 return 对象中的属性(也可以返回一个 Promise 实例,但需要 Suspense 和异步组件的配合)
如下,通过函数修改个人信息
<template>
<h1>姓名:{{name}}h1>
<h1>年龄:{{age}}h1>
<button @click="changeInfo">改变个人信息button>
template>
<script>
export default {
name: 'App',
setup() {
// 数据
let name = "李四"
let age = 18
// 方法
function changeInfo() {
name = "张三"
age = 20
console.log(name + age);
}
return {
name,
age,
changeInfo
}
}
}
script>
发现实际值已经修改了,但页面并没有响应改变

此时需要使用 ref 函数来定义响应的数据
const xxx = ref(initValue)
xxx.value.value,直接:{{xxx}},vue3 自动解析了<script>
import { ref } from 'vue';
export default {
name: 'App',
setup() {
// 数据
let name = ref('李四')
let age = ref(18)
// 方法
function changeInfo() {
name.value = "张三"
age.value = 20
console.log(name, age);
}
return {
name,
age,
changeInfo
}
}
}
</script>
使用了 ref 函数的对象已经是一个 RefImpl 对象(Reference Implement)

假如是对象类型的数据,此时 RefImpl 对象的 value 是一个 Proxy 对象
<template>
<h1>姓名:{{name}}h1>
<h1>年龄:{{age}}h1>
<h2>职业:{{job.type}}h2>
<h2>薪水:{{job.salary}}h2>
<button @click="changeInfo">改变个人信息button>
template>
<script>
import { ref } from 'vue';
export default {
name: 'App',
setup() {
// 数据
let name = ref('李四')
let age = ref(18)
let job = ref({
type: '开发',
salary: '20k'
})
// 方法
function changeInfo() {
job.value.type = '设计'
job.value.salary = '25k'
console.log(job);
console.log(job.value);
}
return {
name,
age,
job,
changeInfo
}
}
}
script>

注意:
Object.defineProperty() 的 get 与 set 完成的const 代理对象 = reactive(源对象),接收一个对象(或数组),返回一个代理对象(Proxy 的实例对象,简称 Proxy 对象)将 3.2 的对象类型示例修改一下,使用 reactive 来定义对象类型数据,如下:
<template>
<h1>姓名:{{name}}h1>
<h1>姓名:{{age}}h1>
<h2>职业:{{job.type}}h2>
<h2>薪水:{{job.salary}}h2>
<button @click="changeInfo">改变个人信息button>
template>
<script>
import { reactive, ref } from 'vue';
export default {
name: 'App',
setup() {
// 数据
let name = ref('李四')
let age = ref(18)
// let job = ref({
// type: '开发',
// salary: '20k'
// })
let job = reactive({
type: '开发',
salary: '20k'
})
// 方法
function changeInfo() {
// job.value.type = '设计'
// job.value.salary = '25k'
job.type = '设计'
job.salary = '25k'
console.log(job);
}
return {
name,
age,
job,
changeInfo
}
}
}
script>

同时还可以响应数组类型以及进行深层次的响应
<template>
<h1>姓名:{{person.name}}h1>
<h1>年龄:{{person.age}}h1>
<h2>职业:{{person.job.type}}h2>
<h2>薪水:{{person.job.salary}}h2>
<h1>爱好:{{person.hobby}}h1>
<button @click="changeInfo">改变个人信息button>
template>
<script>
import { reactive } from 'vue';
export default {
name: 'App',
setup() {
// 数据
let person = reactive({
name: '李四',
age: 18,
job: {
type: '开发',
salary: '20k'
},
hobby: ['看剧', '听歌']
})
// 方法
function changeInfo() {
person.name = '张三'
person.age = 20
person.job.type = '设计'
person.job.salary = '25k'
person.hobby[0] = '学习'
}
return {
person,
changeInfo
}
}
}
script>

实现原理:
Object.defineProperty() 对属性的读取、修改进行拦截(数据劫持),但对新增和删除无法响应式的改变Object.defineProperty(data, 'count', {
configurable: true, // 可配置,即可删除属性
get () { return ... },
set () { 响应... }
})
存在问题:
methods: {
addSex(){
this.person.sex = '女'; // 不生效
this.$set(this.person, 'sex', '女')
Vue.set(this.person, 'sex', '女')
},
deleteName(){
this.$delete(this.person, 'name', '张三')
Vue.delete(this.person, 'name', '张三')
}
},
methods: {
updateHobby(){
this.person.hobby[0] = '学习'; // 不生效
this.$set(this.person.hobby, 0, '学习')
Vue.delete(this.person.hobby, 0, '学习')
this.person.hobby.splice(0, 1, '学习')
}
},
实现原理:
通过Proxy(代理):拦截对象中任意属性的变化,包括属性值的读写、属性的添加、属性的删除等
通过Reflect(反射):对源对象的属性进行操作
<body>
<script type="text/javascript">
let person = {
name: '张三',
age: 18
}
// 模拟 Vue3中实现响应式
const p = new Proxy(person, {
// 读取时调用
get(target, prop) {
console.log(target, prop);
return target[prop]
},
// 修改或新增时调用
set(target, prop, value) {
console.log(target, prop, value);
target[prop] = value;
},
// 删除时调用
deleteProperty(target, prop) {
console.log(target, prop);
return delete target[prop];
}
})
script>
body>
对原对象的代理对象进行操作,而代理对象操作原对象

可以把上述操作交给 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'
MDN 文档中描述的 Proxy 与 Reflect:
<template>
<h1>姓名:{{person.name}}h1>
<h1>年龄:{{person.age}}h1>
<h1 v-show="person.sex">性别:{{person.sex}}h1>
<h2>职业:{{person.job.type}}h2>
<h2>薪水:{{person.job.salary}}h2>
<button @click="changeInfo">改变个人信息button>
template>
<script>
import { reactive } from 'vue';
export default {
name: 'App',
setup() {
// 数据
let person = reactive({
name: '李四',
age: 18,
job: {
type: '开发',
salary: '20k'
},
hobby: ['看剧', '听歌']
})
// 方法
function changeInfo() {
person.sex = '男'
delete person.name
person.hobby[0] = '学习'
}
return {
person,
changeInfo
}
}
}
script>

1、 从定义数据角度对比
2、从原理角度对比
Object.defineProperty() 的 get 与 set 来实现响应式(数据劫持)3、从使用角度对比
.value,读取数据时模板中直接读取不需要 .value.value在 beforeCreate 之前执行一次,this 是 undefined
props:值为对象,包含:组件外部传递过来,且组件内部声明接收了的属性
App.vue 组件,传值进 Demo.vue
<template>
<Demo
msg="你好"
school="无"
/>
template>
<script>
import Demo from './components/Demo.vue';
export default {
name: 'App',
components: {
Demo
}
}
script>
Demo.vue 组件,接收 App.vue 传进来的值
<template>
<h1>个人信息h1>
<h2>姓名:{{person.name}}h2>
<h2>年龄:{{person.age}}h2>
template>
<script>
import { reactive } from 'vue'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Demo',
props: ['msg', 'school'],
setup(props) {
console.log(props);
// 数据
let person = reactive({
name: '张三',
age: 18
})
return {
person
}
}
}
script>
<style>
style>

context:上下文对象
this.$attrsthis.$slotsthis.$emitApp.vue,传值进 Demo.vue,同时传入自定义事件以及插槽
<template>
<Demo
msg="你好"
school="无"
@hello="showMsg"
>
<template v-slot:te>
<span>测试span>
template>
Demo>
template>
<script>
import Demo from './components/Demo.vue';
export default {
name: 'App',
setup() {
function showMsg(value) {
alert(`触发,参数是${value}`)
}
return {
showMsg
}
},
components: {
Demo
}
}
script>
Demo.vue 组件,props 接收 App.vue 传进来的值,未接收的值在 attrs 里,emits 接收自定义事件,假如未写接收会报警告,但不影响使用,emit 触发 App.vue 里的自定义事件。插槽直接使用
<template>
<h1>个人信息h1>
<h2>姓名:{{person.name}}h2>
<h2>年龄:{{person.age}}h2>
<button @click="hello">测试触发事件button>
<slot name="te">slot>
template>
<script>
import { reactive } from 'vue'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Demo',
props: ['msg'],
emits: ['hello'],
setup(props, context) {
console.log(props);
console.log(context.attrs);
console.log(context.emit);
console.log(context.slots);
// 数据
let person = reactive({
name: '张三',
age: 18
})
// 方法
function hello() {
context.emit('hello', 666);
}
return {
person,
hello
}
}
}
script>
<style>
style>

第一种写法:与 Vue2.x 中 computed 配置功能一致
<script>
import { reactive, computed } from 'vue'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Demo',
computed: {
fullName() {
return this.person.firstName + '-' + this.person.lastName;
}
},
setup() {
// 数据
let person = reactive({
firstName: '张',
lastName: '三',
})
return {
person,
}
}
}
</script>
第二种写法,如下:
<template>
<h1>个人信息h1>
姓:<input
type="text"
v-model="person.firstName"
/> <br>
名:<input
type="text"
v-model="person.lastName"
/> <br>
<span>全名: {{person.fullName}} span> <br>
全名:<input
type="text"
v-model="person.fullName"
>
template>
<script>
import { reactive, computed } from 'vue'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Demo',
setup() {
// 数据
let person = reactive({
firstName: '张',
lastName: '三',
})
// 计算属性-简写(只读,不考虑计算属性被修改的情况)
person.fullName = computed(() => {
return person.firstName + '-' + person.lastName;
})
// 计算属性-完整写法(考虑读和写)
person.fullName = computed({
get() {
return person.firstName + '-' + person.lastName;
},
set(value) {
let nameArr = value.split('-');
person.firstName = nameArr[0]
person.lastName = nameArr[1]
}
})
return {
person,
}
}
}
script>
<style>
style>

第一种写法:与 Vue2.x 中 computed 配置功能一致
<script>
import { ref } from 'vue'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Demo',
watch: {
sum(newValue, oldValue) {
console.log('sum 的值变化了', newValue, oldValue);
}
},
// watch: {
// sum: {
// immediate: true, // 立即监视,一进来就会监视一下
// deep: true,
// handler(newValue, oldValue) {
// console.log('sum 的值变化了', newValue, oldValue);
// }
// }
// },
setup() {
// 数据
let sum = ref(0)
return {
sum,
}
}
}
</script>

第二种写法,如下:
1、监视 ref 定义的数据
<template>
<h1>当前求和为:{{sum}} h1>
<button @click="sum++">点击 + 1button>
<hr>
<h1>当前信息为:{{msg}} h1>
<button @click="msg += '!'">修改信息button>
template>
<script>
import { ref, watch } from 'vue'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Demo',
setup() {
// 数据
let sum = ref(0)
let msg = ref('你好')
// 情况一,监视 ref 所定义的一个响应式数据
// watch(sum, (newValue, oldValue) => {
// console.log('sum 变了', newValue, oldValue);
// }, {immediate: true, deep: true})
// 情况二,监视 ref 所定义的多个响应式数据
watch([sum, msg], (newValue, oldValue) => {
console.log(newValue, oldValue);
}, { immediate: true, deep: true })
return {
sum,
msg
}
}
}
script>
<style>
style>

2、监视 reactive 定义的数据
<template>
<h1>当前姓名为:{{person.name}} h1>
<button @click="person.name += '~'">修改姓名button>
<hr>
<h1>当前年龄为:{{person.age}} h1>
<button @click="person.age ++">修改年龄button>
<hr>
<h2>职业:{{person.job.type}} h2>
<h2>薪酬:{{person.job.salary}} h2>
<button @click="person.job.type += '!'">修改职业button>
template>
<script>
import { reactive, watch } from 'vue'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Demo',
setup() {
// 数据
let person = reactive({
name: '张三',
age: 18,
job: {
type: '开发',
salary: '25k'
}
})
// 情况一,监视 reactive 所定义的一个响应式数据的全部属性
// watch 监视的是 reactive 定义的响应式数据,则无法正确获得 oldValue!!
// 若watch 监视的是 reactive 定义的响应式数据,则强制开启了深度监视(deep 配置无效)
// watch(person, (newValue, oldValue) => {
// console.log('person变化', newValue, oldValue);
// }, { deep: false })
// 情况二:监视 reactive 所定义的一个响应式数据的某个属性
// watch(() => person.age, (newValue, oldValue) => {
// console.log('person 的age 变化了', newValue, oldValue);
// })
// 情况三:监视 reactive 所定义的一个响应式数据的某些属性
watch([() => person.age, () => person.name], (newValue, oldValue) => {
console.log('person 的age 变化了', newValue, oldValue);
})
// 特殊情况:监视的是 reactive 定义的属性中的某个对象属性,所以 deep 配置有效
watch(() => person.job, (newValue, oldValue) => {
console.log('person 的age 变化了', newValue, oldValue);
}, { deep: true })
return {
person
}
}
}
script>
<style>
style>

<script>
import { reactive, watchEffect } from 'vue'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Demo',
setup() {
// 数据
let sum = ref(0)
let person = reactive({
name: '张三',
age: 18,
job: {
type: '开发',
salary: '25k'
}
})
// watchEffect 所指定的回调中用到的数据只要发生变化,则直接重新执行回调
watchEffect(() => {
const x1 = person.name
const x2 = person.job.type
console.log('watchEffect 配置的回调执行了');
})
return {
person
}
}
}
</script>

Vue3.0 中可以继续使用 Vue2.x 中的生命周期钩子,但有两个被更名:
App.vue
<template>
<button @click="isShowDemo = !isShowDemo">显示/隐藏Demobutton>
<Demo v-if="isShowDemo" />
template>
<script>
import Demo from './components/Demo.vue';
import { ref } from 'vue'
export default {
name: 'App',
setup() {
let isShowDemo = ref(true)
return {
isShowDemo
}
},
components: {
Demo
}
}
script>
Demo.vue
<template>
<h1>Demoh1>
template>
<script>
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Demo',
beforeCreate() {
console.log('---beforeCreate---');
},
created() {
console.log('---created---');
},
beforeMount() {
console.log('---beforeMount---');
},
mounted() {
console.log('---mounted---');
},
beforeUnmount() {
console.log('---beforeUnmount---');
},
unmounted() {
console.log('---unmounted--');
},
}
script>


Vue3.0 也提供了 Composition API 形式的生命周期钩子,与 Vue2.x 中钩子对应关系如下:
setup()setup()<template>
<h1>Demoh1>
template>
<script>
import { onBeforeMount, onMounted, onBeforeUpdate, onUpdated, onBeforeUnmount, onUnmounted } from 'vue'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Demo',
setup() {
console.log('---setup()---');
onBeforeMount(() => {
console.log('---onBeforeMount---')
})
onMounted(() => {
console.log('---onMounted---');
})
onBeforeUpdate(() => {
console.log('---onBeforeUpdate---');
})
onUpdated(() => {
console.log('---onUpdated---');
})
onBeforeUnmount(() => {
console.log('---onBeforeUnmount---');
})
onUnmounted(() => {
console.log('---onUnmounted---');
})
}
}
script>


hook 本质是一个函数,把 setup 函数中使用的 Composition API 进行了封装。类似于 Vue2.x中的 mixin。可以复用代码,让 setup 中的逻辑更清楚易懂
如下,获取当前鼠标的位置
<template>
<h1>当前点击时鼠标的坐标为:X:{{point.x}},y:{{point.y}} h1>
template>
<script>
import { reactive, onMounted } from 'vue'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Demo',
setup() {
let point = reactive({
x: 0,
y: 0
})
onMounted(() => {
window.addEventListener('click', (event) => {
point.x = event.pageX
point.y = event.pageY
})
})
return {
point
}
}
}
script>

此时给 window 加了一个点击事件,只要点击页面就会获取页面的鼠标位置给 point,假如该组件卸载了,也还是会触发

可以在组件卸载后,移除该点击事件。移除事件时需要传入移除的是哪个事件,所以将该点击事件单独写成一个函数
<script>
import { reactive, onMounted, onBeforeUnmount } from 'vue'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Demo',
setup() {
let point = reactive({
x: 0,
y: 0
})
function savePoint(event) {
point.x = event.pageX
point.y = event.pageY
console.log(point.x, point.y);
}
onMounted(() => {
window.addEventListener('click', savePoint)
})
onBeforeUnmount(() => {
window.removeEventListener('click', savePoint)
})
return {
point
}
}
}
</script>
假如有另一个组件也想用该功能,复用这块代码,就可以将该功能相关的数据和函数抽离出来,形成一个 hook 函数

import { reactive, onMounted, onBeforeUnmount} from 'vue';
export default function() {
let point = reactive({
x: 0,
y: 0
})
function savePoint(event) {
point.x = event.pageX
point.y = event.pageY
console.log(point.x, point.y);
}
onMounted(() => {
window.addEventListener('click', savePoint)
})
onBeforeUnmount(() => {
window.removeEventListener('click', savePoint)
})
return point
}
使用时引入即可
<template>
<h1>当前点击时鼠标的坐标为:X:{{point.x}},y:{{point.y}} h1>
template>
<script>
import usePoint from '../hooks/usePoint'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Demo',
setup() {
let point = usePoint()
return {
point
}
}
}
script>

创建一个 ref 对象,其 value 值指向另一个对象中的某个属性。
const name = toRef(person, 'name')用于要将响应式对象中的某个属性单独提供给外部使用时,如下,将 person 对象里的name、age 等属性单独提供给外部使用
<template>
<h1>姓名: {{name}} h1>
<h1>年龄: {{age}} h1>
<h1>薪资: {{salary}} h1>
<button @click="name += '~'">修改姓名button>
<button @click="age ++">修改年龄button>
<button @click="salary ++">修改薪资button>
template>
<script>
import { reactive, toRef } from '@vue/reactivity'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Demo',
setup() {
let person = reactive({
name: '张三',
age: 18,
job: {
j1: {
salary: 20
}
}
})
return {
// name: ref(person, 'name')
name: toRef(person, 'name'),
age: toRef(person, 'age'),
salary: toRef(person.job.j1, 'salary')
}
}
}
script>

具有响应式效果,且此时 toRef 这里操纵的数据,就是原先的 person对象里的数据,修改即同步修改 person 对象里的对应的值
假如直接使用 ref 来转换,如:ref(person.name),操纵的是这个用 person 对象的 name 属性值新建的 ref 对象,而不再与原本的 person 对象的 name 有联系

toRefs 与 toRef 功能一致,但可以批量创建多个 ref 对象,即把整个对象都抛出去,语法:toRefs(person),但只能定位到第外层的属性
<h1>姓名: {{name}} </h1>
<h1>年龄: {{age}} </h1>
<h1>薪资: {{job.j1.salary}} </h1>
...toRefs(person)
// name: toRef(person, 'name'),
// age: toRef(person, 'age'),
// salary: toRef(person.job.j1, 'salary')
let person = shallowReactive ({
name: '张三',
age: 18,
job: {
j1: {
salary: 20
}
}
})
什么时候使用?
let person = reactive ({
name: '张三',
age: 18,
job: {
j1: {
salary: 20
}
}
})
person = readonly(person)
应用于不希望数据被修改时
toRaw
使用场景:用于读取响应式对象对应的普通对象,对这个普通对象的所有操作,不会引起页面更新
const p = toRaw(person)
markRaw
应用场景:
person.car = markRaw(person)
创建一个自定义的 ref,并对其依赖项跟踪和更新触发进行显式控制
如下例,实现防抖效果
<template>
<input
type="text"
v-model="keyWord"
/>
<h1> {{keyWord}} h1>
template>
<script>
import { customRef } from 'vue'
export default {
name: 'App',
setup() {
// 自定义一个 ref
function myRef(value, delay) {
let timer;
return customRef((track, trigger) => {
return {
get() {
// 通知 Vue 追踪数据变化(提前约定)
track();
return value;
},
set(newValue) {
value = newValue;
clearTimeout(timer);
timer = setTimeout(() => {
// 通知 Vue 去重新解析模板(调用一下 get() 方法)
trigger();
}, delay);
}
}
})
}
// let keyWord = ref('hello') // 使用 Vue 提供的 ref
let keyWord = myRef('hello', '500') // 使用自定义的 ref
return {
keyWord,
myRef
}
},
}
script>


实现祖与后代组件间通信,父组件有一个 provide 选项来提供数据,后代组件有一个 inject 选项来开始使用这些数据
如下,祖组件 App.vue
<template>
<div class="app">
<h1> App 组件(祖),{{name}}---{{price}} h1>
<Child />
div>
template>
<script>
import { reactive, toRefs } from '@vue/reactivity'
import Child from './components/Child.vue'
import { provide } from '@vue/runtime-core'
export default {
name: 'App',
setup() {
let car = reactive({
name: '奔驰',
price: '40w'
})
provide('car', car)
return {
...toRefs(car)
}
},
components: {
Child
}
}
script>
子组件,Child.vue
<template>
<div class="child">
<h1> Child组件(子)h1>
<Son />
div>
template>
<script>
import Son from './Son.vue'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Child',
components: {
Son
}
}
script>
孙组件,Son.vue
<template>
<div class="son">
<h1> Son组件(孙),{{car.name}}---{{car.price}}h1>
div>
template>
<script>
import { inject } from '@vue/runtime-core'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Son',
setup() {
let car = inject('car')
return {
car
}
}
}
script>

let car = reactive({name: '奔驰', price: '40w'})
let sum = ref(0)
let car2 = readonly(car)
console.log(isReactive(car))
console.log(isRef(sum))
console.log(isReadonly(car2))
console.log(isReadonly(isProxy))
Vue2 使用的传统 Options API 中,新增或者修改一个需求,就需要分别在 data,methods,computed 里修改


可以更加优雅的组织我们的代码,函数。让相关功能的代码更加有序的组织在一起


在 Vue2 中:组件必须有一个根标签。在 Vue3 中:组件可以没有根标签,内部会将多个标签包含在一个 Fragment 虚拟元素中,可以减少标签层级, 减小内存占用
<template>
<h1>111h1>
<h1>222h1>
template>

能够将组件 html 结构移动到指定位置
如下,直接打开弹窗会撑开组件及其父组件的高度

组件 Son.vue
<template>
<div class="son">
<h1> Son 组件(孙)h1>
<Dialog />
div>
template>
<script>
import Dialog from './Dialog.vue'
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Son',
components: {
Dialog
}
}
script>
组件 Dialog.vue
<template>
<button @click="isShow = true"> 点击弹窗 button>
<div
v-if="isShow"
class="dialog"
>
<h1>弹窗内容h1>
<h1>弹窗内容h1>
<h1>弹窗内容h1>
<button @click="isShow = false">关闭弹窗button>
div>
template>
<script>
import { ref } from '@vue/reactivity';
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Dialog',
setup() {
let isShow = ref(false);
return {
isShow
}
}
}
script>
利用 Teleport 将弹窗移动到 body 上
<template>
<button @click="isShow = true"> 点击弹窗 button>
<teleport to='body'>
<div
v-if="isShow"
class="mask"
>
<div class="dialog">
<h1>弹窗内容h1>
<h1>弹窗内容h1>
<h1>弹窗内容h1>
<button @click="isShow = false">关闭弹窗button>
div>
div>
teleport>
template>
<script>
import { ref } from '@vue/reactivity';
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Dialog',
setup() {
let isShow = ref(false);
return {
isShow
}
}
}
script>
<style>
.mask {
position: absolute;
top: 0;
bottom: 0;
left: 0;
right: 0;
background-color: rgba(0, 0, 0, 0.5);
}
.dialog {
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
text-align: center;
background-color: green;
width: 300px;
height: 300px;
}
style>

等待异步组件时渲染一些额外内容,让应用有更好的用户体验
静态引入
<template>
<div class="app">
<h1> App 组件(祖)h1>
<Child />
div>
template>
<script>
import Child from './components/Child.vue' // 静态引入
export default {
name: 'App',
components: {
Child
}
}
script>
将网速调慢,祖组件和后代组件是一起出来的

异步引入
<template>
<div class="app">
<h1> App 组件(祖)h1>
<Child />
div>
template>
<script>
// import Child from './components/Child.vue' //静态引入
import { defineAsyncComponent } from '@vue/runtime-core' // 静态引入
const Child = defineAsyncComponent(() => import('./components/Child.vue')) // 异步引入
export default {
name: 'App',
components: {
Child
}
}
script>
网速慢的情况下,会先出现祖组件,再出现后代组件

此时有个问题,假如 Child 组件没有加载出来,其所在的位置是空的,并不知道到底有没有内容。使用 Suspense 解决如下:
<template>
<div class="app">
<h1> App 组件(祖)h1>
<Suspense>
<template v-slot:default>
<Child />
template>
<template v-slot:fallback>
<h3>稍等,加载中...h3>
template>
Suspense>
div>
template>
<script>
// import Child from './components/Child.vue' //静态引入
import { defineAsyncComponent } from '@vue/runtime-core' // 静态引入
const Child = defineAsyncComponent(() => import('./components/Child.vue')) // 异步引入
export default {
name: 'App',
components: {
Child
}
}
script>

使用了异步引入后,该组件是一个异步组件,则 setup() 可以用 async 修一个异步函数
<script>
import { ref } from 'vue';
export default {
// eslint-disable-next-line vue/multi-word-component-names
name: 'Child',
async setup() {
let sum = ref(0)
let p = new Promise((resove, reject) => {
setTimeout(() => {
resove(sum)
}, 3000);
})
return await p;
}
}
</script>
之前是通过把网速调慢来实现等待 Child 组件出现的效果,使用异步函数之后,网速正常也能让 Child 组件等待后才出现

Vue 2.x 有许多全局 API 和配置,例如:注册全局组件、注册全局指令等
//注册全局组件
Vue.component('MyButton', {
data: () => ({
count: 0
}),
template: ''
})
//注册全局指令
Vue.directive('focus', {
inserted: el => el.focus()
}
Vue3.0 中对这些 API 做出了调整,将全局的 API,即:Vue.xxx 调整到应用实例(app)上
| 2.x 全局 API(Vue) | 3.x 实例 API (app) |
|---|---|
| app.config.xxxx | app.config.xxxx |
| Vue.config.productionTip | 移除 |
| Vue.component | app.component |
| Vue.directive | app.directive |
| Vue.mixin | app.mixin |
| Vue.use | app.use |
| Vue.prototype | app.config.globalProperties |
// data { }
data() { }
.v-enter,
.v-leave-to {
opacity: 0;
}
.v-leave,
.v-enter-to {
opacity: 1;
}
.v-enter-from,
.v-leave-to {
opacity: 0;
}
.v-leave-from,
.v-enter-to {
opacity: 1;
}
// @keyup.13
// Vue.config.keyCodes.enter = 13
<my-component
v-on:close="handleComponentEvent"
v-on:click="handleNativeClickEvent"
/>
<script>
export default {
emits: ['close']
}
</script>