分享 15 个比较常见的问题,基本都贴出对应文档地址,还请多看文档~ 因此主要分几个方面总结:
文档地址:https://v3.cn.vuejs.org/guide/composition-api-lifecycle-hooks.html
Vue2.x 和 Vue3.x 生命周期方法的变化蛮大的,先看看:
2.x 生命周期 | 3.x 生命周期 | 执行时间说明 |
beforeCreate | setup | 组件创建前执行 |
created | setup | 组件创建后执行 |
beforeMount | onBeforeMount | 组件挂载到节点上之前执行 |
mounted | onMounted | 组件挂载完成后执行 |
beforeUpdate | onBeforeUpdate | 组件更新之前执行 |
updated | onUpdated | 组件更新完成之后执行 |
beforeDestroy | onBeforeUnmount | 组件卸载之前执行 |
destroyed | onUnmounted | 组件卸载完成后执行 |
errorCaptured | onErrorCaptured | 当捕获一个来自子孙组件的异常时激活钩子函数 |
目前 Vue3.x 依然支持 Vue2.x 的生命周期,但不建议混搭使用,前期可以先使用 2.x 的生命周期,后面尽量使用 3.x 的生命周期开发。
由于我使用都是 script-srtup 模式,所以都是直接使用 Vue3.x 的生命周期函数:
复制
- // A.vue
- <script setup lang="ts">
- import { ref, onMounted } from "vue";
- let count = ref<number>(0);
-
- onMounted(() => {
- count.value = 1;
- })
- </script>
每个钩子的执行时机点,也可以看看文档:https://v3.cn.vuejs.org/guide/instance.html#生命周期图示
文档地址:https://v3.cn.vuejs.org/api/sfc-script-setup.html#defineexpose
这里主要介绍父组件如何去获取子组件内部定义的变量,关于父子组件通信,可以看文档介绍比较详细:https://v3.cn.vuejs.org/guide/component-basics.html
我们可以使用 全局编译器宏 的 defineExpose 宏,将子组件中需要暴露给父组件获取的参数,通过 {key: vlaue} 方式作为参数即可,父组件通过模版 ref 方式获取子组件实例,就能获取到对应值:
复制
- // 子组件
- <script setup>
- let name = ref("pingan8787")
- defineExpose({ name }); // 显式暴露的数据,父组件才可以获取
- </script>
-
- // 父组件
- <Chlid ref="child"></Chlid>
- <script setup>
- let child = ref(null)
- child.value.name //获取子组件中 name 的值为 pingan8787
- </script>
注意:
definedProps 文档:https://v3.cn.vuejs.org/api/sfc-script-setup.html#defineprops-%E5%92%8C-defineemitswithDefaults 文档:https://v3.cn.vuejs.org/api/sfc-script-setup.html#%E4%BB%85%E9%99%90-typescript-%E7%9A%84%E5%8A%9F%E8%83%BD
前面介绍 script-setup 模式提供的 4 个 全局编译器宏 ,还没有详细介绍,这一节介绍 defineProps 和 withDefaults 。使用 defineProps 宏可以用来定义组件的入参,使用如下:
复制
- <script setup lang="ts">
- let props = defineProps<{
- schema: AttrsValueObject;
- modelValue: any;
- }>();
- </script>
这里只定义 props 属性中的 schema 和 modelValue 两个属性的类型, defineProps 的这种声明的不足之处在于,它没有提供设置 props 默认值的方式。
其实我们可以通过 withDefaults 这个宏来实现:
复制
- <script setup lang="ts">
- let props = withDefaults(
- defineProps<{
- schema: AttrsValueObject;
- modelValue: any;
- }>(),
- {
- schema: [],
- modelValue: ''
- }
- );
- </script>
withDefaults 辅助函数提供了对默认值的类型检查,并确保返回的 props 的类型删除了已声明默认值的属性的可选标志。
文档地址:https://v3.cn.vuejs.org/guide/migration/global-api.html#vue-prototype-%E6%9B%BF%E6%8D%A2%E4%B8%BA-config-globalproperties
在 Vue2.x 中我们可以通过 Vue.prototype 添加全局属性 property。但是在 Vue3.x 中需要将 Vue.prototype 替换为 config.globalProperties 配置:
复制
- // Vue2.x
- Vue.prototype.$api = axios;
- Vue.prototype.$eventBus = eventBus;
-
- // Vue3.x
- const app = createApp({})
- app.config.globalProperties.$api = axios;
- app.config.globalProperties.$eventBus = eventBus;
使用时需要先通过 vue 提供的 getCurrentInstance 方法获取实例对象:
- // A.vue
-
- <script setup lang="ts">
- import { ref, onMounted, getCurrentInstance } from "vue";
-
- onMounted(() => {
- const instance = <any>getCurrentInstance();
- const { $api, $eventBus } = instance.appContext.config.globalProperties;
- // do something
- })
- </script>
其中 instance 内容输出如下:
文档地址:https://v3.cn.vuejs.org/guide/migration/v-model.html
当我们在使用 v-model 指令的时候,实际上 v-bind 和 v-on 组合的简写,Vue2.x 和 Vue3.x 又存在差异。
- <ChildComponent v-model="pageTitle" />
-
- <ChildComponent :value="pageTitle" @input="pageTitle = $event" />
在子组件中,如果要对某一个属性进行双向数据绑定,只要通过 this.$emit('update:myPropName', newValue) 就能更新其 v-model 绑定的值。
- <ChildComponent v-model="pageTitle" />
-
-
- <ChildComponent :modelValue="pageTitle" @update:modelValue="pageTitle = $event"/>
script-setup 模式下就不能使用 this.$emit 去派发更新事件,毕竟没有 this ,这时候需要使用前面有介绍到的 defineProps、defineEmits 两个宏来实现:
- // 子组件 child.vue
- // 文档:https://v3.cn.vuejs.org/api/sfc-script-setup.html#defineprops-%E5%92%8C-defineemits
- <script setup lang="ts">
- import { ref, onMounted, watch } from "vue";
- const emit = defineEmits(['update:modelValue']); // 定义需要派发的事件名称
-
- let curValue = ref('');
- let props = withDefaults(defineProps<{
- modelValue: string;
- }>(), {
- modelValue: '',
- })
-
- onMounted(() => {
- // 先将 v-model 传入的 modelValue 保存
- curValue.value = props.modelValue;
- })
-
- watch(curValue, (newVal, oldVal) => {
- // 当 curValue 变化,则通过 emit 派发更新
- emit('update:modelValue', newVal)
- })
-
- </script>
-
- <template>
- <div></div>
- </template>
-
- <style lang="scss" scoped></style>
父组件使用的时候就很简单:
- // 父组件 father.vue
-
- <script setup lang="ts">
- import { ref, onMounted, watch } from "vue";
- let curValue = ref('');
-
- watch(curValue, (newVal, oldVal) => {
- console.log('[curValue 发生变化]', newVal)
- })
- </script>
-
- <template>
- <Child v-model='curValue'></Child>
- </template>
-
- <style lang="scss" scoped></style>
文档地址:https://v3.cn.vuejs.org/api/application-config.html#errorhandler
Vue3.x 对于一些开发过程中的异常,做了更友好的提示警告,比如下面这个提示:
这样能够更清楚的告知异常的出处,可以看出大概是 复制 这时候就能看到输出内容如下: 一下子就清楚很多。当然,该配置项也可以用来集成错误追踪服务 Sentry 和 Bugsnag。推荐阅读:Vue3 如何实现全局异常处理? 当我们在控制台输出 ref 声明的变量时。 复制 会看到控制台输出了一个 RefImpl 对象: 看起来很不直观。我们都知道,要获取和修改 ref 声明的变量的值,需要通过 .value 来获取,所以你也可以: 复制 这里还有另一种方式,就是在控制台的设置面板中开启 「 Enable custom formatters 」选项。 这时候你会发现,控制台输出的 ref 的格式发生变化了: 更加清晰直观了。 这个方法是我在《Vue.js 设计与实现》中发现的,但在文档也没有找到相关介绍,如果有朋友发现了,欢迎告知~ 1. Vite 动态导入的使用问题 文档地址:https://cn.vitejs.dev/guide/features.html#glob-import 使用 webpack 的同学应该都知道,在 webpack 中可以通过 require.context 动态导入文件: 复制 在 Vite 中,我们可以使用这两个方法来动态导入文件: 该方法匹配到的文件默认是 懒加载 ,通过 动态导入 实现,构建时会 分离独立的 chunk ,是 异步导入 ,返回的是 Promise,需要做异步操作,使用方式如下: 复制 该方法是 直接导入所有模块 ,并且是 同步导入 ,返回结果直接通过 for...in 循环就可以操作,使用方式如下: 复制 如果仅仅使用异步导入 Vue3 组件,也可以直接使用 Vue3 defineAsyncComponent API 来加载: 复制 文档地址:https://cn.vitejs.dev/config/#resolve-alias 当项目比较复杂的时候,经常需要配置 alias 路径别名来简化一些代码: 复制 在 Vite 中配置也很简单,只需要在 vite.config.ts 的 resolve.alias 中配置即可: 复制 如果使用的是 TypeScript 时,编辑器会提示路径不存在的警告:warning:,这时候可以在 tsconfig.json 中添加 compilerOptions.paths 的配置: 复制 文档地址:https://cn.vitejs.dev/config/#css-preprocessoroptions 当我们需要使用 scss 配置的主题变量(如 $primary )、mixin方法(如 @mixin lines )等时,如: 复制 我们可以将 scss 主题配置文件,配置在 vite.config.ts 的 css.preprocessorOptions.scss.additionalData 中: 复制 如果不想使用 scss 配置文件,也可以直接写成 scss 代码: 复制 文档地址:https://router.vuejs.org/zh/guide/advanced/composition-api.html 由于在 script-setup 模式下,没有 this 可以使用,就不能直接通过 this.$router 或 this.$route 来获取路由参数和跳转路由。当我们需要获取路由参数时,就可以使用 vue-router 提供的 useRoute 方法来获取,使用如下: 复制 如果要做路由跳转,就可以使用 useRouter 方法的返回值去跳转: 复制 文档地址:https://pinia.vuejs.org/core-concepts/#using-the-store 当我们解构出 store 的变量后,再修改 store 上该变量的值,视图没有更新: 复制 这时候点击按钮触发 changeName 事件后,视图上的 name 并没有变化。这是因为 store 是个 reactive 对象,当进行解构后,会破坏它的响应性。所以我们不能直接进行解构。这种情况就可以使用 Pinia 提供 storeToRefs 工具方法,使用起来也很简单,只需要将需要解构的对象通过 storeToRefs 方法包裹,其他逻辑不变: 复制 这样再修改其值,变更马上更新视图了。 (1)按照官网给的方案,目前有三种方式修改: 通过 store.属性名 赋值修改单笔数据的状态; 这个方法就是前面一节使用的: 复制 (2)通过 $patch 方法修改多笔数据的状态; 文档地址:https://pinia.vuejs.org/api/interfaces/pinia._StoreWithState.html#patch 当我们需要同时修改多笔数据的状态时,如果还是按照上面方法,可能要这么写: 复制 上面这么写也没什么问题,但是 Pinia 官网已经说明,使用 $patch 的效率会更高,性能更好,所以在修改多笔数据时,更推荐使用 $patch ,使用方式也很简单: 复制 (3)通过 action 方法修改多笔数据的状态; 也可以在 store 中定义 actions 的一个方法来更新: 复制 使用时: 复制 这三种方式都能更新 Pinia 中 store 的数据状态。 项目新安装的 element-plus 在开发阶段都是正常,没有提示任何警告,但是在打包过程中,控制台输出下面警告内容: 在官方 issues 中查阅很久:https://github.com/element-plus/element-plus/issues/3219。 尝试在 vite.config.ts 中配置 charset: false ,结果也是无效: 复制 最后在官方的 issues 中找到处理方法: 复制 文档地址:https://element-plus.gitee.io/zh-CN/guide/i18n.html#%E5%85%A8%E5%B1%80%E9%85%8D%E7%BD%AE 默认 elemnt-plus 的组件是英文状态: 我们可以通过引入中文语言包,并添加到 ElementPlus 配置中来切换成中文: 复制 这时候就能看到 ElementPlus 里面组件的文本变成中文了。 以上是我最近从入门到实战 Vue3 全家桶的 3 个项目后总结避坑经验,其实很多都是文档中有介绍的,只是刚开始不熟悉。也希望大伙多看看文档咯~ Vue3 script-setup 模式确实越写越香。7. 观察 ref 的数据不直观,不方便
console.log('[测试 ref]', count.value);
二、Vite
2. Vite 配置 alias 类型别名
import Home from '@/views/Home.vue'
3. Vite 配置全局 scss
三、VueRouter
1. script-setup 模式下获取路由参数
四、Pinia
1. store 解构的变量修改后没有更新
2. Pinia 修改数据状态的方式
五、Element Plus
1. element-plus 打包时 @charset 警告
2. 中文语言包配置
总结