# vue3
**Repository Path**: yunfeng918/vue3
## Basic Information
- **Project Name**: vue3
- **Description**: No description available
- **Primary Language**: Unknown
- **License**: Not specified
- **Default Branch**: master
- **Homepage**: None
- **GVP Project**: No
## Statistics
- **Stars**: 0
- **Forks**: 0
- **Created**: 2023-10-26
- **Last Updated**: 2023-11-11
## Categories & Tags
**Categories**: Uncategorized
**Tags**: Vue, vue3
## README
# vue-test-v3
# 创建 VUE 3 工程
# 常用 Compostion API
## 1 初始 setup
1. 理解:Vue3.0 中一个新的配置项,值为一个函数。
2. setup 是所有 Composition API(组合 API)的入口。
3. 组件中所用到的:数据、方法等等,均要配置在 setup 中。
4. setup 函数的两种返回值:
1. 若返回一个对象,则对象中的属性、方法,在模板中可以直接使用。(常用)
2. 若返回一个渲染函数,则可以自定义渲染内容。(了解)
5. 注意点:
1. 尽量不要与 Vue2.x 配置混用
1. Vue2.x 配置(data、methos、computed...)中可以访问 setup 中的属性、方法。
2. setup 中不能访问 Vue2.x 配置(data、methos、computed...)。
3. 如果有重名,setup 优先。
2. setup 不能是一个 async 函数,因为返回值不再是 return 的对象,而是 promise,模板看不到 return 对象中的属性。注意:后期也可以返回一个 Promise 实例,但需要 Suspense 和异步组件的配合才能实现。
## 2 ref 函数
1. 作用:定义一个响应式数据。
2. 语法:`const xxx = ref(initValue)`
1. 创建一个包含响应式数据的引用对象(reference 对象,简称 ref 对象)。
2. JS 中操作数据:`xxx.value`
3. 模板中操作数据:`
{{xxx}}
`, 不需要`.value`
3. 备注:
1. 接收的数据可以是:基本类型、对象类型。
2. 基表类型的数据:响应式依然是由`Object.defineProperty()`的 get、set 完成的
3. 对象类型的数据:内部“求助”了 Vue3.0 中的一个新函数——`reactive`函数。
## 3 reactive 函数
1. 作用:定义一个对象类型的响应式数据(**基本类型不要用它**,否则 Vue 无法监听)。
2. 语法:`const 代理对象= reactive(源对象)`,接收一个对象(或数组),返回一个代理对象(Proxy 的实例对象,简称 procy 对象)
3. `reactive`定义的响应式数据是“深层次的”。
4. 内部基于 ES6 的`Proxy`实现,通过代理对象操作源对象内部数据。
5. reactive 与 ref 的区别:
1. ref 用来定义:**响应式**的**基本类型数据**。
2. reactive 用来定义:**响应式**的**对象类型数据(或数组类型数据)**。
## 4 Vue3.0 中的响应式原理
### 1. Vue2 的响应式原理
1. 实现原理:
1. 对象类型:通过`Object.defineProperty()`对属性的读、写、删进行拦截(数据劫持)。`Object.defineProperty(proxyPerson, key, {get(){}, set(val){}});`
2. 数组类型:通过重写更新数组的一系列方法来实现拦截。(对数组的方法进行重写)
2. 存在问题:
1. 新增属性、删除属性,默认无法触发响应式,可以手动去触发(`vm.$set`、`vm.$delete`)。
2. 直接通过下标修改数组,默认无法触发响应式,可以手动去触发(`vm.$set`)。
### 2. Vue3 的响应式原理
1. 实现原理:
1. 通过`Proxy`对象来对数据进行拦截(拦截对象属性的读、改、删、增)。
2. 通过`Reflect`对象来对数据进行操作。
3. MDN 文档中描述的 Proxy 和 Reflect 对象:
1. Proxy: https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Proxy
2. Reflect: https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Reflect
2. 实现步骤:
```JavaScript
// 原数据
let person = {
name: '张三',
age: 18,
}
// 代理数据
// 使用ES6的Proxy代理person对象
// 读取proxyPerson的属性时,会自动执行get方法
// 修改或添加proxyPerson的属性时,会自动执行set方法
// 删除proxyPerson的属性时,会自动执行deleteProperty方法
const proxyPerson = new Proxy(person, {
get(target, propName) {
console.log('读取了:' + propName + " 可以实现响应式");
return Reflect.get(target, propName);
},
set(target, propName, value) {
console.log('添加 或 修改了:' + propName + " 可以实现响应式");
Reflect.set(target, propName, value);
},
deleteProperty(target, propName) {
console.log('删除了:' + propName + " 可以实现响应式");
return Reflect.deleteProperty(target, propName);
}
});
```
## 5 reactive 对比 ref
1. 从定义数据角度对比:
1. ref 用来定义:**基本类型数据**
2. reactive 用来定义:**对象(或数组)类型数据**
3. 备注:ref 也可以用来定义**对象(或数组)类型数据**,它内部会自动通过`reactive`转换成**代理对象**。
2. 从原理角度对比:
1. ref 通过`Object.defineProperty()`的`get`与`set`来实现响应式(数据劫持)。
2. reactive 通过使用`Proxy`来实现响应式(数据劫持), 并通过`Reflect`操作`源对象`内部的属性。
3. 从使用角度对比:
1. ref 定义的数据:操作数据**需要**`.value`, 读取数据时模板中**不需要**`.value`。
2. reactive 定义的数据:操作数据与读取数据:**均不需要**`.value`。
## 6 setup 注意点
1. setup 执行的时机:在 beforeCreate 之前执行一次,此时组件实例尚未创建,this 是 undefined。
2. setup 的参数:
1. props: 值为对象,包含:组件外部传递过来且组件内部声明接收了的属性
2. context: 上下文对象,里面有 attrs、slots、emit 等属性。
1. attrs: 值为对象,包含:组件外部传递过来,但没有在 props 配置中声明的属性,相当于 Vue2 的`this.$attrs`。
2. slots: 收到的插槽内容,相当于 Vue2 的`this.$slots`。
3. emit: 分类自定义事件的函数,相当于 Vue2 的`this.$emit`。
## 7 计算属性与监视
1. 计算属性:computed
1. 与 vue2 中 computed 配置功能一致
2. 写法:
```JavaScript
import {computed} from 'vue';
setup() {
// 数据
let person = reactive({
firstName: '张',
lastName: '三',
age: 18
});
// 计算属性-简写(计算属性只读,不可修改)
// person.fullName = computed(function () {
// return person.firstName + person.lastName
// });
// 计算属性-完整写法
person.fullName = computed({
get() {
return person.firstName + "-" + person.lastName;
},
set(value) {
const nameArr = value.split("-");
person.firstName = nameArr[0];
person.lastName = nameArr[1];
}
});
return { person }
}
```
2. 监视属性:watch
1. 与 Vue2 中的 watch 配置功能一致
2. 三个小坑:
1. 监视 reactive 定义的响应式数据时,oldValue 无法正确获取,因为 oldValue 是在 getter 中获取的,而此时 getter 已经执行了,所以是获取不到正确的旧值
2. 监视 reactive 定义的响应式数据时,强制开启了深度监视(deep 配置无效)
3. 监视 reactive 定义的响应式数据中的某个对象属性时,deep 配置有效
3. watchEffect 函数
1. watch 的套路是:既要指明要监视的属性,也要指明监视的回调。
2. watchEffect 的套路是:不用指明要监视的属性,监视的回调中用到哪个属性就监视哪个属性。
3. watchEffect 有点像 computed,
1. 但 computed 注重的计算出来的值(回调函数的返回值),所以必须要写返回值。
2. 而 watchEffect 更注重的是过程(回调函数的函数体),所以不用写返回值。
```JavaScript
watchEffect(() => {
let x = num.value;
let y = person.job.job1.salary;
console.log("Vue3 watchEffect监视执行了, 该函数用到了谁,就会监视谁", x, y);
});
```
## 8 Vue3 生命周期

1. Vue3 中可以继续使用 Vue2 中的生命周期钩子函数,但有两个改名了:
- beforeDestroy 改名为 beforeUnmount
- destroyed 改名为 unmounted
2. Vue3 也提供了 Composition API 形式的生命周期钩子, 与 Vue2 中钩子对应关系如下:
- beforeCreate ==> setup()
- created ==> setup()
- beforeMount ==> onBeforeMount
- mounted ==> onMounted
- beforeUpdate ==> onBeforeUpdate
- updated ==> onUpdated
- beforeUnmount ==> onBeforeUnmount
- unmounted ==> onUnmounted
## 9 自定义 hook 函数
1. 什么是 hook:本质是一个函数,吧 setup 函数中使用的 Composition API 进行了封装。
2. 类似于 Vue2 中的 mixin。
3. 自定义 hook 的优势: 复用代码,让 setup 中的逻辑更清楚易懂。
## 10 toRef
1. 作用:创建一个 ref 对象,其 value 值指向另一个对象中的某个属性。
2. 语法:const name = toRef(person, 'name');
3. 应用:要将响应式对象中的某个属性单独提供给外部使用时。
4. 扩展:toRefs 与 toRef 功能一致,但可以批量创建多个 ref 对象。语法:`toRefs(person)`
# 三 其它 Composition API
## 1 shallowReactive 与 ShallowRef
1. shallowReactive:只处理对象最外层属性的响应式(浅响应式)。
2. shallowRef:只处理基本数据类型的响应式,不进行对象的响应式处理。
3. 应用场景:
1. 如果有一个对象数据,结构比较深,但变化时只是外层属性变化,则使用 shallowReactive。
2. 如果有一个对象数据,后续功能不会修改对象中的属性,而是生新的对象来替换,则使用 shallowRef。
## 2 readonly 与 shallowReadonly
1. readonly:让一个响应式数据变为只读的(深只读)。
2. shallowReadonly:让一个响应式数据变为只读的(浅只读)。
3. 应用场景:不希望数据被修改时。
## 3 toRaw 与 markRaw
1. toRaw:
1. 作用:讲一个 reactive 生成的**响应式对象**转为**原始对象**。
2. 应用场景:用于读取响应式对象对应的普通对象,对这个普通对象的所有操作,不会引起页面更新。
2. markRaw:
1. 作用:标记一个对象,使其永远不会再成为响应式对象。
2. 应用场景:
1. 有些值不应被设置为响应式的,例如复杂的第三方类库等。
2. 当渲染具有不可变数据源的大列表时,跳过响应式转换可以提高性能。
## 4 customRef
1. 作用:创建一个自定义的 ref,并对其依赖项跟踪和更新触发进行显示控制。
2. 实现防抖效果:
```javaScript
setup() {
function myRef(value, delay) {
return customRef((track, trigger) => {
let timer;
return {
get() {
console.log("从myRef中读取了数据 ", value, track);
track(); // 通知Vue追踪这个value的变化
return value;
},
set(newValue) {
clearTimeout(timer);
console.log("把myRef中的数据修改了 ", newValue, trigger);
timer = setTimeout(() => {
value = newValue;
trigger(); // 通知Vue重新解析模板,Vue重新解析模板调用get方法
}, delay)
}
}
});
}
let keyword = myRef("", 500);
return {
keyword
}
},
```
## 5 provide 与 inject
1. 作用:实现**祖与后代组件间**通信。
2. 套路:祖组件有一个 `provide` 选项来提供数据,后代组件有一个 `inject` 选项来开始使用这些数据。
3. 具体写法:
1. 祖组件中:
```JavaScript
```
2. 后代组件中:
```JavaScript
```
## 6 响应式数据的判断
1. isRef:检查一个值是否为一个 ref 对象。
2. isReactive:检查一个对象是否为一个由 `reactive` 创建的响应式对象。
3. isReadonly:检查一个对象是否为一个由 `readonly` 创建的只读响应式对象。
4. isProxy:检查一个对象是否由 `reactive` 或 `readonly` 创建的代理。
# 四 Composition API 的优势
1. Options API 存在的问题:新增或者修改一个需求,就需要分别在 data、methods、computed、watch 里修改。
2. Composition API 优势:可以更加优雅的组织我们的代码在一个 hook 函数里,让相关功能的代码更加紧密有序的在一起。
# 五 新的组件
## 1 Fragment
1. 在 Vue2 中: 组件必须有一个根标签
2. 在 Vue3 中: 组件可以没有根标签, 内部会将多个标签封装到 Fragment 中
3. 好处: 减少标签层级, 减小内存占用
## 2 Teleport
1. 什么是 Teleport:Teleport 是一种能够将我们的`组件html结构`移动到指定位置的技术。
```html
```
## 3 Suspense
1. 等他异步组件时渲染一些额外内容,让应用有更好的用户体验
2. 使用步骤:
1. 异步引入组件
```js
import { defineAsyncComponent } from "vue";
const Child = defineAsyncComponent(() => import("./components/Child"));
```
2. 使用 Suspense 包裹组件,并配置号 default 和 fallback
```html
Loading...
```
3. setup 可以使用 async 修饰,返回一个 Promise 对象
```js
async setup() {
let num = ref(100);
let p = new Promise((resolve) => {
setTimeout(() => {
resolve({ num });
}, 3000);
});
return await p;
}
```
# 六 其他
## 1 全局 API 的转移
1. Vue2.x 有许多全局 API, 例如:注册全局组件,注册全局指令等。
```js
Vue.component('MyComp',{ el:"app", data: ()=>({ msg: 'hello'})})
Vue.directive('myFocus', {inserted: function (el, binding) { el.focus()});
```
2. Vue3.0 中这些 API 做出了调整:将全局的 API,调整到应用实例(app)上了
| 2.x 全局 API | 3.x 实例 API |
| ------------ | ------------ |
| Vue.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 |
## 2 其他更改
1. data 选项应始终被声明为一个函数
2. 过度类名的更改
1. Vue2.x 写法
```css
.fade-enter,
.fade-leave-to {
opacity: 0;
}
.v-leave,
.v-leave-to {
opacity: 1;
}
```
2. Vue3.x 写法
```css
.fade-enter-from,
.fade-leave-to {
opacity: 0;
}
.v-leave-from,
.v-leave-to {
opacity: 1;
}
```
3. **移除**keyCode 作用 v-on 的修饰符,同时也不再支持 config.keyCodes
4. **移除**v-on.native 修饰符
1. 父组件中绑定事件
```html
```
2. 子组件中申明自定义事件
```js
export default {
emits: ["myClose"],
};
```
5. **移除**过滤器(filter)
> 过滤器虽然这看起来很方便,但它需要一个自定义语法,打破大括号内表达式是“只是 JavaScript”的假设,这不仅有学习成本,而且有实现成本,建议用方法调用或计算属性去替换过滤器