[{"data":1,"prerenderedAt":-1},["ShallowReactive",2],{"site-config":3,"article-vue3快速上手":22,"footer-socials":51,"comments-vue3快速上手":52},{"site_title":4,"site_subtitle":5,"home_intro":6,"avatar":7,"seo_keywords":8,"seo_description":9,"site_subtitle_highlight":10,"home_description":11,"about_name":12,"about_intro":13,"code_comment_2":14,"code_log":15,"code_skills":16,"code_goal":17,"code_comment_1":18,"meteor_density":19,"meteor_max_count":19,"meteor_enabled":20,"meteor_speed":21},"ShineGoldYao","架构代码，","全栈开发者 \u002F 开源爱好者 \u002F 技术探索者","https:\u002F\u002Fforuda.gitee.com\u002Favatar\u002F1762402862010015318\u002F16382196_yaoxingjin_1762402861.png!avatar200","ShiGoldYao,技术博客,全栈开发","专注于前沿技术分享与开源项目展示的个人技术博客","书写未来。","大家好，我是 ShiGoldYao。一名全栈学习与技术爱好者。在这里，我分享关于现代 Web 开发、技术框架学习记录以及极客生活的深度思考。","ShiGoldYao","我是一名全栈技术学习者，对构建高性能、可扩展的现代 Web 应用充满热情。过去一年里，我从初识互联网前后端开发，到逐步沉淀技术体系，始终保持着对前端、后端与工程化的持续探索。\n\n我坚信 “代码如诗”。除了日常学习与项目实践，我也会花大量时间关注开源社区，尝试摸索 WebAssembly、Rust 等前沿技术在浏览器端的更多可能，不断挑战性能与体验的边界。\n\n生活里的我并不只有代码：闲暇时会打打永劫无间，享受博弈与操作的快感；也喜欢打乒乓球，在运动中放松自己；当然，最幸福的时光，还是和女朋友一起慢慢生活、认真恋爱。","\u002F\u002F 🚀 开启学习之旅","","Vue , TypeScript , Nest , Mysql","成为优秀的前端全栈工程师","\u002F\u002F 欢迎来到我的技术世界","3","true","5",{"id":23,"title":24,"slug":25,"coverUrl":15,"summary":15,"content":26,"htmlContent":27,"categoryId":28,"viewCount":29,"likeCount":30,"isTop":30,"isPublish":31,"seoKeywords":15,"seoDescription":15,"publishTime":32,"createTime":33,"updateTime":34,"deleteTime":35,"category":36,"tags":42},"12","1. Vue3简介","vue3快速上手","# 1. Vue3简介\n- 2020年9月18日，`Vue.js`发布版`3.0`版本，代号：`One Piece`（n\n- 经历了：[4800+次提交](https:\u002F\u002Fgithub.com\u002Fvuejs\u002Fcore\u002Fcommits\u002Fmain)、[40+个RFC](https:\u002F\u002Fgithub.com\u002Fvuejs\u002Frfcs\u002Ftree\u002Fmaster\u002Factive-rfcs)、[600+次PR](https:\u002F\u002Fgithub.com\u002Fvuejs\u002Fvue-next\u002Fpulls?q=is%3Apr+is%3Amerged+-author%3Aapp%2Fdependabot-preview+)、[300+贡献者](https:\u002F\u002Fgithub.com\u002Fvuejs\u002Fcore\u002Fgraphs\u002Fcontributors)\n- 官方发版地址：[Release v3.0.0 One Piece · vuejs\u002Fcore](https:\u002F\u002Fgithub.com\u002Fvuejs\u002Fcore\u002Freleases\u002Ftag\u002Fv3.0.0)\n- 截止2023年10月，最新的公开版本为：`3.3.4`\n\n  \u003Cimg src=\"images\u002F1695089947298-161c1b47-eb86-42fb-b1f8-d6a4fcab8ee2.png\" alt=\"image.png\" style=\"zoom:30%;\" \u002F> \n\n## 1.1. 【性能的提升】\n\n- 打包大小减少`41%`。\n\n- 初次渲染快`55%`, 更新渲染快`133%`。\n\n- 内存减少`54%`。\n\n  \n## 1.2.【 源码的升级】\n\n- 使用`Proxy`代替`defineProperty`实现响应式。\n\n- 重写虚拟`DOM`的实现和`Tree-Shaking`。\n\n  \n## 1.3. 【拥抱TypeScript】\n\n- `Vue3`可以更好的支持`TypeScript`。\n\n  \n## 1.4. 【新的特性】\n\n1. `Composition API`（组合`API`）：\n   - `setup`\n   - `ref`与`reactive`\n   - `computed`与`watch`\n   \n     ......\n   \n2. 新的内置组件：\n   - `Fragment`\n   - `Teleport`\n   - `Suspense`\n\n     ......\n\n3. 其他改变：\n   - 新的生命周期钩子\n   - `data` 选项应始终被声明为一个函数\n   - 移除`keyCode`支持作为` v-on` 的修饰符\n\n     ......\n\n\n\n# 2. 创建Vue3工程\n\n## 2.1. 【基于 vue-cli 创建】\n点击查看[官方文档](https:\u002F\u002Fcli.vuejs.org\u002Fzh\u002Fguide\u002Fcreating-a-project.html#vue-create)\n\n> 备注：目前`vue-cli`已处于维护模式，官方推荐基于 `Vite` 创建项目。\n\n```powershell\n## 查看@vue\u002Fcli版本，确保@vue\u002Fcli版本在4.5.0以上\nvue --version\n\n## 安装或者升级你的@vue\u002Fcli \nnpm install -g @vue\u002Fcli\n\n## 执行创建命令\nvue create vue_test\n\n##  随后选择3.x\n##  Choose a version of Vue.js that you want to start the project with (Use arrow keys)\n##  > 3.x\n##    2.x\n\n## 启动\ncd vue_test\nnpm run serve\n```\n\n---\n\n## 2.2. 【基于 vite 创建】(推荐)\n`vite` 是新一代前端构建工具，官网地址：[https:\u002F\u002Fvitejs.cn](https:\u002F\u002Fvitejs.cn\u002F)，`vite`的优势如下：\n\n- 轻量快速的热重载（`HMR`），能实现极速的服务启动。\n- 对 `TypeScript`、`JSX`、`CSS` 等支持开箱即用。\n- 真正的按需编译，不再等待整个应用编译完成。\n- `webpack`构建 与 `vite`构建对比图如下：\n\u003Cimg src=\"images\u002F1683167182037-71c78210-8217-4e7d-9a83-e463035efbbe.png\" alt=\"webpack构建\" title=\"webpack构建\" style=\"zoom:20%;box-shadow:0 0 10px black\" \u002F>\t\u003Cimg src=\"images\u002F1683167204081-582dc237-72bc-499e-9589-2cdfd452e62f.png\" alt=\"vite构建\" title=\"vite构建\" style=\"zoom: 20%;box-shadow:0 0 10px black\" \u002F>\n* 具体操作如下（点击查看[官方文档](https:\u002F\u002Fcn.vuejs.org\u002Fguide\u002Fquick-start.html#creating-a-vue-application)）\n\n```powershell\n## 1.创建命令\nnpm create vue@latest\n\n## 2.具体配置\n## 配置项目名称\n√ Project name: vue3_test\n## 是否添加TypeScript支持\n√ Add TypeScript?  Yes\n## 是否添加JSX支持\n√ Add JSX Support?  No\n## 是否添加路由环境\n√ Add Vue Router for Single Page Application development?  No\n## 是否添加pinia环境\n√ Add Pinia for state management?  No\n## 是否添加单元测试\n√ Add Vitest for Unit Testing?  No\n## 是否添加端到端测试方案\n√ Add an End-to-End Testing Solution? » No\n## 是否添加ESLint语法检查\n√ Add ESLint for code quality?  Yes\n## 是否添加Prettiert代码格式化\n√ Add Prettier for code formatting?  No\n```\n自己动手编写一个App组件\n\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"app\">\n    \u003Ch1>你好啊！\u003C\u002Fh1>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript lang=\"ts\">\n  export default {\n    name:'App' \u002F\u002F组件名\n  }\n\u003C\u002Fscript>\n\n\u003Cstyle>\n  .app {\n    background-color: #ddd;\n    box-shadow: 0 0 10px;\n    border-radius: 10px;\n    padding: 20px;\n  }\n\u003C\u002Fstyle>\n```\n\n安装官方推荐的`vscode`插件：\n\n\u003Cimg src=\"images\u002Fvolar.png\" alt=\"Snipaste_2023-10-08_20-46-34\" style=\"zoom:50%;\" \u002F> \n\n\u003Cimg src=\"images\u002Fimage-20231218085906380.png\" alt=\"image-20231218085906380\" style=\"zoom:42%;\" \u002F> \n\n总结：\n\n- `Vite` 项目中，`index.html` 是项目的入口文件，在项目最外层。\n- 加载`index.html`后，`Vite` 解析 `\u003Cscript type=\"module\" src=\"xxx\">` 指向的`JavaScript`。\n- `Vue3`**中是通过 **`createApp` 函数创建一个应用实例。\n## 2.3. 【一个简单的效果】\n\n`Vue3`向下兼容`Vue2`语法，且`Vue3`中的模板中可以没有根标签\n\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"person\">\n    \u003Ch2>姓名：{{name}}\u003C\u002Fh2>\n    \u003Ch2>年龄：{{age}}\u003C\u002Fh2>\n    \u003Cbutton @click=\"changeName\">修改名字\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeAge\">年龄+1\u003C\u002Fbutton>\n    \u003Cbutton @click=\"showTel\">点我查看联系方式\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript lang=\"ts\">\n  export default {\n    name:'App',\n    data() {\n      return {\n        name:'张三',\n        age:18,\n        tel:'13888888888'\n      }\n    },\n    methods:{\n      changeName(){\n        this.name = 'zhang-san'\n      },\n      changeAge(){\n        this.age += 1\n      },\n      showTel(){\n        alert(this.tel)\n      }\n    },\n  }\n\u003C\u002Fscript>\n```\n\n\n# 3. Vue3核心语法\n## 3.1.  【OptionsAPI 与 CompositionAPI】\n\n- `Vue2`的`API`设计是`Options`（配置）风格的。\n- `Vue3`的`API`设计是`Composition`（组合）风格的。\n###  Options API 的弊端\n\n`Options`类型的 `API`，数据、方法、计算属性等，是分散在：`data`、`methods`、`computed`中的，若想新增或者修改一个需求，就需要分别修改：`data`、`methods`、`computed`，不便于维护和复用。\n\n\u003Cimg src=\"images\u002F1696662197101-55d2b251-f6e5-47f4-b3f1-d8531bbf9279.gif\" alt=\"1.gif\" style=\"zoom:70%;border-radius:20px\" \u002F>\u003Cimg src=\"images\u002F1696662200734-1bad8249-d7a2-423e-a3c3-ab4c110628be.gif\" alt=\"2.gif\" style=\"zoom:70%;border-radius:20px\" \u002F>\n\n### Composition API 的优势\n\n可以用函数的方式，更加优雅的组织代码，让相关功能的代码更加有序的组织在一起。\n\n\u003Cimg src=\"images\u002F1696662249851-db6403a1-acb5-481a-88e0-e1e34d2ef53a.gif\" alt=\"3.gif\" style=\"height:300px;border-radius:10px\"  \u002F>\u003Cimg src=\"images\u002F1696662256560-7239b9f9-a770-43c1-9386-6cc12ef1e9c0.gif\" alt=\"4.gif\" style=\"height:300px;border-radius:10px\"  \u002F>\n\n> 说明：以上四张动图原创作者：大帅老猿\n\n## 3.2. 【拉开序幕的 setup】\n### setup 概述\n`setup`是`Vue3`中一个新的配置项，值是一个函数，它是 `Composition API` **“表演的舞台**_**”**_，组件中所用到的：数据、方法、计算属性、监视......等等，均配置在`setup`中。\n\n特点如下：\n\n- `setup`函数返回的对象中的内容，可直接在模板中使用。\n- `setup`中访问`this`是`undefined`。\n- `setup`函数会在`beforeCreate`之前调用，它是“领先”所有钩子执行的。\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"person\">\n    \u003Ch2>姓名：{{name}}\u003C\u002Fh2>\n    \u003Ch2>年龄：{{age}}\u003C\u002Fh2>\n    \u003Cbutton @click=\"changeName\">修改名字\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeAge\">年龄+1\u003C\u002Fbutton>\n    \u003Cbutton @click=\"showTel\">点我查看联系方式\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript lang=\"ts\">\n  export default {\n    name:'Person',\n    setup(){\n      \u002F\u002F 数据，原来写在data中（注意：此时的name、age、tel数据都不是响应式数据）\n      let name = '张三'\n      let age = 18\n      let tel = '13888888888'\n\n      \u002F\u002F 方法，原来写在methods中\n      function changeName(){\n        name = 'zhang-san' \u002F\u002F注意：此时这么修改name页面是不变化的\n        console.log(name)\n      }\n      function changeAge(){\n        age += 1 \u002F\u002F注意：此时这么修改age页面是不变化的\n        console.log(age)\n      }\n      function showTel(){\n        alert(tel)\n      }\n\n      \u002F\u002F 返回一个对象，对象中的内容，模板中可以直接使用\n      return {name,age,tel,changeName,changeAge,showTel}\n    }\n  }\n\u003C\u002Fscript>\n```\n### setup 的返回值\n\n- 若返回一个**对象**：则对象中的：属性、方法等，在模板中均可以直接使用**（重点关注）。**\n- 若返回一个**函数**：则可以自定义渲染内容，代码如下：\n```jsx\nsetup(){\n  return ()=> '你好啊！'\n}\n```\n### setup 与 Options API 的关系\n\n- `Vue2` 的配置（`data`、`methos`......）中**可以访问到** `setup`中的属性、方法。（Vue2和Vue3可以共存）\n- 但在`setup`中**不能访问到**`Vue2`的配置（`data`、`methos`......）。\n- 如果与`Vue2`冲突，则`setup`优先。\n### setup 语法糖\n`setup`函数有一个语法糖，这个语法糖，可以让我们把`setup`独立出去，代码如下：\n\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"person\">\n    \u003Ch2>姓名：{{name}}\u003C\u002Fh2>\n    \u003Ch2>年龄：{{age}}\u003C\u002Fh2>\n    \u003Cbutton @click=\"changName\">修改名字\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changAge\">年龄+1\u003C\u002Fbutton>\n    \u003Cbutton @click=\"showTel\">点我查看联系方式\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript lang=\"ts\">\n  export default {\n    name:'Person',\n  }\n\u003C\u002Fscript>\n\n\u003C!-- 下面的写法是setup语法糖 -->\n\u003Cscript setup lang=\"ts\">\n  console.log(this) \u002F\u002Fundefined\n  \n  \u002F\u002F 数据（注意：此时的name、age、tel都不是响应式数据）\n  let name = '张三'\n  let age = 18\n  let tel = '13888888888'\n\n  \u002F\u002F 方法\n  function changName(){\n    name = '李四'\u002F\u002F注意：此时这么修改name页面是不变化的\n  }\n  function changAge(){\n    console.log(age)\n    age += 1 \u002F\u002F注意：此时这么修改age页面是不变化的\n  }\n  function showTel(){\n    alert(tel)\n  }\n\u003C\u002Fscript>\n```\n扩展：上述代码，还需要编写一个不写`setup`的`script`标签，去指定组件名字，比较麻烦，我们可以借助`vite`中的插件简化\n\n1. 第一步：`npm i vite-plugin-vue-setup-extend -D`\n2. 第二步：`vite.config.ts`\n```jsx\nimport { defineConfig } from 'vite'\nimport VueSetupExtend from 'vite-plugin-vue-setup-extend'\n\nexport default defineConfig({\n  plugins: [ VueSetupExtend() ]\n})\n```\n\n3. 第三步：`\u003Cscript setup lang=\"ts\" name=\"Person\">`\n## 3.3. 【ref 创建：基本类型的响应式数据】\n\n- **作用：**定义响应式变量。\n- **语法：**`let xxx = ref(初始值)`。\n- **返回值：**一个`RefImpl`的实例对象，简称`ref对象`或`ref`，`ref`对象的`value`**属性是响应式的**。\n- **注意点：**\n   - `JS`中操作数据需要：`xxx.value`，但模板中不需要`.value`，直接使用即可。\n   - 对于`let name = ref('张三')`来说，`name`不是响应式的，`name.value`是响应式的。\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"person\">\n    \u003Ch2>姓名：{{name}}\u003C\u002Fh2>\n    \u003Ch2>年龄：{{age}}\u003C\u002Fh2>\n    \u003Cbutton @click=\"changeName\">修改名字\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeAge\">年龄+1\u003C\u002Fbutton>\n    \u003Cbutton @click=\"showTel\">点我查看联系方式\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript setup lang=\"ts\" name=\"Person\">\n  import {ref} from 'vue'\n  \u002F\u002F name和age是一个RefImpl的实例对象，简称ref对象，它们的value属性是响应式的。\n  let name = ref('张三')\n  let age = ref(18)\n  \u002F\u002F tel就是一个普通的字符串，不是响应式的\n  let tel = '13888888888'\n\n  function changeName(){\n    \u002F\u002F JS中操作ref对象时候需要.value\n    name.value = '李四'\n    console.log(name.value)\n\n    \u002F\u002F 注意：name不是响应式的，name.value是响应式的，所以如下代码并不会引起页面的更新。\n    \u002F\u002F name = ref('zhang-san')\n  }\n  function changeAge(){\n    \u002F\u002F JS中操作ref对象时候需要.value\n    age.value += 1 \n    console.log(age.value)\n  }\n  function showTel(){\n    alert(tel)\n  }\n\u003C\u002Fscript>\n```\n## 3.4. 【reactive 创建：对象类型的响应式数据】\n\n- **作用：**定义一个**响应式对象**（基本类型不要用它，要用`ref`，否则报错）\n- **语法：**`let 响应式对象= reactive(源对象)`。\n- **返回值：**一个`Proxy`的实例对象，简称：响应式对象。\n- **注意点：**`reactive`定义的响应式数据是“深层次”的。\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"person\">\n    \u003Ch2>汽车信息：一台{{ car.brand }}汽车，价值{{ car.price }}万\u003C\u002Fh2>\n    \u003Ch2>游戏列表：\u003C\u002Fh2>\n    \u003Cul>\n      \u003Cli v-for=\"g in games\" :key=\"g.id\">{{ g.name }}\u003C\u002Fli>\n    \u003C\u002Ful>\n    \u003Ch2>测试：{{obj.a.b.c.d}}\u003C\u002Fh2>\n    \u003Cbutton @click=\"changeCarPrice\">修改汽车价格\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeFirstGame\">修改第一游戏\u003C\u002Fbutton>\n    \u003Cbutton @click=\"test\">测试\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript lang=\"ts\" setup name=\"Person\">\nimport { reactive } from 'vue'\n\n\u002F\u002F 数据\nlet car = reactive({ brand: '奔驰', price: 100 })\nlet games = reactive([\n  { id: 'ahsgdyfa01', name: '英雄联盟' },\n  { id: 'ahsgdyfa02', name: '王者荣耀' },\n  { id: 'ahsgdyfa03', name: '原神' }\n])\nlet obj = reactive({\n  a:{\n    b:{\n      c:{\n        d:666\n      }\n    }\n  }\n})\n\nfunction changeCarPrice() {\n  car.price += 10\n}\nfunction changeFirstGame() {\n  games[0].name = '流星蝴蝶剑'\n}\nfunction test(){\n  obj.a.b.c.d = 999\n}\n\u003C\u002Fscript>\n```\n## 3.5. 【ref 创建：对象类型的响应式数据】\n\n- 其实`ref`接收的数据可以是：**基本类型**、**对象类型**。\n- 若`ref`接收的是对象类型，内部其实也是调用了`reactive`函数。\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"person\">\n    \u003Ch2>汽车信息：一台{{ car.brand }}汽车，价值{{ car.price }}万\u003C\u002Fh2>\n    \u003Ch2>游戏列表：\u003C\u002Fh2>\n    \u003Cul>\n      \u003Cli v-for=\"g in games\" :key=\"g.id\">{{ g.name }}\u003C\u002Fli>\n    \u003C\u002Ful>\n    \u003Ch2>测试：{{obj.a.b.c.d}}\u003C\u002Fh2>\n    \u003Cbutton @click=\"changeCarPrice\">修改汽车价格\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeFirstGame\">修改第一游戏\u003C\u002Fbutton>\n    \u003Cbutton @click=\"test\">测试\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript lang=\"ts\" setup name=\"Person\">\nimport { ref } from 'vue'\n\n\u002F\u002F 数据\nlet car = ref({ brand: '奔驰', price: 100 })\nlet games = ref([\n  { id: 'ahsgdyfa01', name: '英雄联盟' },\n  { id: 'ahsgdyfa02', name: '王者荣耀' },\n  { id: 'ahsgdyfa03', name: '原神' }\n])\nlet obj = ref({\n  a:{\n    b:{\n      c:{\n        d:666\n      }\n    }\n  }\n})\n\nconsole.log(car)\n\nfunction changeCarPrice() {\n  car.value.price += 10\n}\nfunction changeFirstGame() {\n  games.value[0].name = '流星蝴蝶剑'\n}\nfunction test(){\n  obj.value.a.b.c.d = 999\n}\n\u003C\u002Fscript>\n```\n## 3.6. 【ref 对比 reactive】\n宏观角度看：\n\n> 1. `ref`用来定义：**基本类型数据**、**对象类型数据**；\n>\n> 2. `reactive`用来定义：**对象类型数据**。\n\n- 区别：\n\n> 1. `ref`创建的变量必须使用`.value`（可以使用`Official`插件自动添加`.value`）。\n>\n>    ![image-20260110235114221](images\u002Fimage-20260110235114221.png)\n>\n>    ![image-20260110235122904](images\u002Fimage-20260110235122904.png)\n>\n> 2. `reactive`重新分配一个新对象，会**失去**响应式（可以使用`Object.assign`去整体替换）。\n\n- 使用原则：\n> 1. 若需要一个基本类型的响应式数据，必须使用`ref`。\n> 2. 若需要一个响应式对象，层级不深，`ref`、`reactive`都可以。\n> 3. 若需要一个响应式对象，且层级较深，推荐使用`reactive`。\n\n## 3.7. 【toRefs 与 toRef】\n\n- 作用：将一个响应式对象中的每一个属性，转换为`ref`对象。\n- 备注：`toRefs`与`toRef`功能一致，但`toRefs`可以批量转换。\n- 语法如下：\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"person\">\n    \u003Ch2>姓名：{{person.name}}\u003C\u002Fh2>\n    \u003Ch2>年龄：{{person.age}}\u003C\u002Fh2>\n    \u003Ch2>性别：{{person.gender}}\u003C\u002Fh2>\n    \u003Cbutton @click=\"changeName\">修改名字\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeAge\">修改年龄\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeGender\">修改性别\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript lang=\"ts\" setup name=\"Person\">\n  import {ref,reactive,toRefs,toRef} from 'vue'\n\n  \u002F\u002F 数据\n  let person = reactive({name:'张三', age:18, gender:'男'})\n\t\n  \u002F\u002F 通过toRefs将person对象中的n个属性批量取出，且依然保持响应式的能力\n  let {name,gender} =  toRefs(person)\n\t\n  \u002F\u002F 通过toRef将person对象中的gender属性取出，且依然保持响应式的能力\n  let age = toRef(person,'age')\n\n  \u002F\u002F 方法\n  function changeName(){\n    name.value += '~'\n  }\n  function changeAge(){\n    age.value += 1\n  }\n  function changeGender(){\n    gender.value = '女'\n  }\n\u003C\u002Fscript>\n```\n## 3.8. 【computed】\n\n作用：根据已有数据计算出新数据（和`Vue2`中的`computed`作用一致）。\n\n\u003Cimg src=\"images\u002Fcomputed.gif\" style=\"zoom:20%;\" \u002F>  \n\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"person\">\n    姓：\u003Cinput type=\"text\" v-model=\"firstName\"> \u003Cbr>\n    名：\u003Cinput type=\"text\" v-model=\"lastName\"> \u003Cbr>\n    全名：\u003Cspan>{{fullName}}\u003C\u002Fspan> \u003Cbr>\n    \u003Cbutton @click=\"changeFullName\">全名改为：li-si\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript setup lang=\"ts\" name=\"App\">\n  import {ref,computed} from 'vue'\n\n  let firstName = ref('zhang')\n  let lastName = ref('san')\n\n  \u002F\u002F 计算属性——只读取，不修改\n  \u002F* let fullName = computed(()=>{\n    return firstName.value + '-' + lastName.value\n  }) *\u002F\n\n\n  \u002F\u002F 计算属性——既读取又修改\n  let fullName = computed({\n    \u002F\u002F 读取\n    get(){\n      return firstName.value + '-' + lastName.value\n    },\n    \u002F\u002F 修改\n    set(val){\n      console.log('有人修改了fullName',val)\n      firstName.value = val.split('-')[0]\n      lastName.value = val.split('-')[1]\n    }\n  })\n\n  function changeFullName(){\n    fullName.value = 'li-si'\n  } \n\u003C\u002Fscript>\n```\n## 3.9.【watch】\n\n- 作用：监视数据的变化（和`Vue2`中的`watch`作用一致）\n- 特点：`Vue3`中的`watch`只能监视以下**四种数据**：\n> 1. `ref`定义的数据。\n> 2. `reactive`定义的数据。\n> 3. 函数返回一个值（`getter`函数）。\n> 4. 一个包含上述内容的数组。\n\n我们在`Vue3`中使用`watch`的时候，通常会遇到以下几种情况：\n### * 情况一\n监视`ref`定义的【基本类型】数据：直接写数据名即可，监视的是其`value`值的改变。\n\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"person\">\n    \u003Ch1>情况一：监视【ref】定义的【基本类型】数据\u003C\u002Fh1>\n    \u003Ch2>当前求和为：{{sum}}\u003C\u002Fh2>\n    \u003Cbutton @click=\"changeSum\">点我sum+1\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript lang=\"ts\" setup name=\"Person\">\n  import {ref,watch} from 'vue'\n  \u002F\u002F 数据\n  let sum = ref(0)\n  \u002F\u002F 方法\n  function changeSum(){\n    sum.value += 1\n  }\n  \u002F\u002F 监视，情况一：监视【ref】定义的【基本类型】数据\n  const stopWatch = watch(sum,(newValue,oldValue)=>{\n    console.log('sum变化了',newValue,oldValue)\n    if(newValue >= 10){\n      stopWatch()\n    }\n  })\n\u003C\u002Fscript>\n```\n### * 情况二\n监视`ref`定义的【对象类型】数据：直接写数据名，监视的是对象的【地址值】，若想监视对象内部的数据，要手动开启深度监视。\n\n> 注意：\n>\n> * 若修改的是`ref`定义的对象中的属性，`newValue` 和 `oldValue` 都是新值，因为它们是同一个对象。\n>\n> * 若修改整个`ref`定义的对象，`newValue` 是新值， `oldValue` 是旧值，因为不是同一个对象了。\n\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"person\">\n    \u003Ch1>情况二：监视【ref】定义的【对象类型】数据\u003C\u002Fh1>\n    \u003Ch2>姓名：{{ person.name }}\u003C\u002Fh2>\n    \u003Ch2>年龄：{{ person.age }}\u003C\u002Fh2>\n    \u003Cbutton @click=\"changeName\">修改名字\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeAge\">修改年龄\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changePerson\">修改整个人\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript lang=\"ts\" setup name=\"Person\">\n  import {ref,watch} from 'vue'\n  \u002F\u002F 数据\n  let person = ref({\n    name:'张三',\n    age:18\n  })\n  \u002F\u002F 方法\n  function changeName(){\n    person.value.name += '~'\n  }\n  function changeAge(){\n    person.value.age += 1\n  }\n  function changePerson(){\n    person.value = {name:'李四',age:90}\n  }\n  \u002F* \n    监视，情况一：监视【ref】定义的【对象类型】数据，监视的是对象的地址值，若想监视对象内部属性的变化，需要手动开启深度监视\n    watch的第一个参数是：被监视的数据\n    watch的第二个参数是：监视的回调\n    watch的第三个参数是：配置对象（deep、immediate等等.....） \n  *\u002F\n  watch(person,(newValue,oldValue)=>{\n    console.log('person变化了',newValue,oldValue)\n  },{deep:true})\n  \n\u003C\u002Fscript>\n```\n### *  情况三\n监视`reactive`定义的【对象类型】数据，且默认开启了深度监视。\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"person\">\n    \u003Ch1>情况三：监视【reactive】定义的【对象类型】数据\u003C\u002Fh1>\n    \u003Ch2>姓名：{{ person.name }}\u003C\u002Fh2>\n    \u003Ch2>年龄：{{ person.age }}\u003C\u002Fh2>\n    \u003Cbutton @click=\"changeName\">修改名字\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeAge\">修改年龄\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changePerson\">修改整个人\u003C\u002Fbutton>\n    \u003Chr>\n    \u003Ch2>测试：{{obj.a.b.c}}\u003C\u002Fh2>\n    \u003Cbutton @click=\"test\">修改obj.a.b.c\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript lang=\"ts\" setup name=\"Person\">\n  import {reactive,watch} from 'vue'\n  \u002F\u002F 数据\n  let person = reactive({\n    name:'张三',\n    age:18\n  })\n  let obj = reactive({\n    a:{\n      b:{\n        c:666\n      }\n    }\n  })\n  \u002F\u002F 方法\n  function changeName(){\n    person.name += '~'\n  }\n  function changeAge(){\n    person.age += 1\n  }\n  function changePerson(){\n    Object.assign(person,{name:'李四',age:80})\n  }\n  function test(){\n    obj.a.b.c = 888\n  }\n\n  \u002F\u002F 监视，情况三：监视【reactive】定义的【对象类型】数据，且默认是开启深度监视的\n  watch(person,(newValue,oldValue)=>{\n    console.log('person变化了',newValue,oldValue)\n  })\n  watch(obj,(newValue,oldValue)=>{\n    console.log('Obj变化了',newValue,oldValue)\n  })\n\u003C\u002Fscript>\n```\n### * 情况四\n监视`ref`或`reactive`定义的【对象类型】数据中的**某个属性**，注意点如下：\n\n1. 若该属性值**不是**【对象类型】，需要写成函数形式。\n2. 若该属性值是**依然**是【对象类型】，可直接编，也可写成函数，建议写成函数。\n\n结论：监视的要是对象里的属性，那么最好写函数式，注意点：若是对象监视的是地址值，需要关注对象内部，需要手动开启深度监视。\n\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"person\">\n    \u003Ch1>情况四：监视【ref】或【reactive】定义的【对象类型】数据中的某个属性\u003C\u002Fh1>\n    \u003Ch2>姓名：{{ person.name }}\u003C\u002Fh2>\n    \u003Ch2>年龄：{{ person.age }}\u003C\u002Fh2>\n    \u003Ch2>汽车：{{ person.car.c1 }}、{{ person.car.c2 }}\u003C\u002Fh2>\n    \u003Cbutton @click=\"changeName\">修改名字\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeAge\">修改年龄\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeC1\">修改第一台车\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeC2\">修改第二台车\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeCar\">修改整个车\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript lang=\"ts\" setup name=\"Person\">\n  import {reactive,watch} from 'vue'\n\n  \u002F\u002F 数据\n  let person = reactive({\n    name:'张三',\n    age:18,\n    car:{\n      c1:'奔驰',\n      c2:'宝马'\n    }\n  })\n  \u002F\u002F 方法\n  function changeName(){\n    person.name += '~'\n  }\n  function changeAge(){\n    person.age += 1\n  }\n  function changeC1(){\n    person.car.c1 = '奥迪'\n  }\n  function changeC2(){\n    person.car.c2 = '大众'\n  }\n  function changeCar(){\n    person.car = {c1:'雅迪',c2:'爱玛'}\n  }\n\n  \u002F\u002F 监视，情况四：监视响应式对象中的某个属性，且该属性是基本类型的，要写成函数式\n  \u002F* watch(()=> person.name,(newValue,oldValue)=>{\n    console.log('person.name变化了',newValue,oldValue)\n  }) *\u002F\n\n  \u002F\u002F 监视，情况四：监视响应式对象中的某个属性，且该属性是对象类型的，可以直接写，也能写函数，更推荐写函数\n  watch(()=>person.car,(newValue,oldValue)=>{\n    console.log('person.car变化了',newValue,oldValue)\n  },{deep:true})\n\u003C\u002Fscript>\n```\n### * 情况五\n监视上述的多个数据\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"person\">\n    \u003Ch1>情况五：监视上述的多个数据\u003C\u002Fh1>\n    \u003Ch2>姓名：{{ person.name }}\u003C\u002Fh2>\n    \u003Ch2>年龄：{{ person.age }}\u003C\u002Fh2>\n    \u003Ch2>汽车：{{ person.car.c1 }}、{{ person.car.c2 }}\u003C\u002Fh2>\n    \u003Cbutton @click=\"changeName\">修改名字\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeAge\">修改年龄\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeC1\">修改第一台车\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeC2\">修改第二台车\u003C\u002Fbutton>\n    \u003Cbutton @click=\"changeCar\">修改整个车\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript lang=\"ts\" setup name=\"Person\">\n  import {reactive,watch} from 'vue'\n\n  \u002F\u002F 数据\n  let person = reactive({\n    name:'张三',\n    age:18,\n    car:{\n      c1:'奔驰',\n      c2:'宝马'\n    }\n  })\n  \u002F\u002F 方法\n  function changeName(){\n    person.name += '~'\n  }\n  function changeAge(){\n    person.age += 1\n  }\n  function changeC1(){\n    person.car.c1 = '奥迪'\n  }\n  function changeC2(){\n    person.car.c2 = '大众'\n  }\n  function changeCar(){\n    person.car = {c1:'雅迪',c2:'爱玛'}\n  }\n\n  \u002F\u002F 监视，情况五：监视上述的多个数据\n  watch([()=>person.name,person.car],(newValue,oldValue)=>{\n    console.log('person.car变化了',newValue,oldValue)\n  },{deep:true})\n\n\u003C\u002Fscript>\n```\n## 3.10. 【watchEffect】\n\n* 官网：立即运行一个函数，同时响应式地追踪其依赖，并在依赖更改时重新执行该函数。\n\n* `watch`对比`watchEffect`\n\n  > 1. 都能监听响应式数据的变化，不同的是监听数据变化的方式不同\n  >\n  > 2. `watch`：要明确指出监视的数据\n  >\n  > 3. `watchEffect`：不用明确指出监视的数据（函数中用到哪些属性，那就监视哪些属性）。\n\n* 示例代码：\n\n  ```vue\n  \u003Ctemplate>\n    \u003Cdiv class=\"person\">\n      \u003Ch1>需求：水温达到50℃，或水位达到20cm，则联系服务器\u003C\u002Fh1>\n      \u003Ch2 id=\"demo\">水温：{{temp}}\u003C\u002Fh2>\n      \u003Ch2>水位：{{height}}\u003C\u002Fh2>\n      \u003Cbutton @click=\"changePrice\">水温+1\u003C\u002Fbutton>\n      \u003Cbutton @click=\"changeSum\">水位+10\u003C\u002Fbutton>\n    \u003C\u002Fdiv>\n  \u003C\u002Ftemplate>\n  \n  \u003Cscript lang=\"ts\" setup name=\"Person\">\n    import {ref,watch,watchEffect} from 'vue'\n    \u002F\u002F 数据\n    let temp = ref(0)\n    let height = ref(0)\n  \n    \u002F\u002F 方法\n    function changePrice(){\n      temp.value += 10\n    }\n    function changeSum(){\n      height.value += 1\n    }\n  \n    \u002F\u002F 用watch实现，需要明确的指出要监视：temp、height\n    watch([temp,height],(value)=>{\n      \u002F\u002F 从value中获取最新的temp值、height值\n      const [newTemp,newHeight] = value\n      \u002F\u002F 室温达到50℃，或水位达到20cm，立刻联系服务器\n      if(newTemp >= 50 || newHeight >= 20){\n        console.log('联系服务器')\n      }\n    })\n  \n    \u002F\u002F 用watchEffect实现，不用\n    const stopWtach = watchEffect(()=>{\n      \u002F\u002F 室温达到50℃，或水位达到20cm，立刻联系服务器\n      if(temp.value >= 50 || height.value >= 20){\n        console.log(document.getElementById('demo')?.innerText)\n        console.log('联系服务器')\n      }\n      \u002F\u002F 水温达到100，或水位达到50，取消监视\n      if(temp.value === 100 || height.value === 50){\n        console.log('清理了')\n        stopWtach()\n      }\n    })\n  \u003C\u002Fscript>\n  ```\n  \n  \n\n## 3.11. 【标签的 ref 属性】\n\n作用：用于注册模板引用。\n\n> * 用在普通`DOM`标签上，获取的是`DOM`节点。\n>\n> * 用在组件标签上，获取的是组件实例对象。\n\n用在普通`DOM`标签上：\n\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"person\">\n    \u003Ch1 ref=\"title1\">尚硅谷\u003C\u002Fh1>\n    \u003Ch2 ref=\"title2\">前端\u003C\u002Fh2>\n    \u003Ch3 ref=\"title3\">Vue\u003C\u002Fh3>\n    \u003Cinput type=\"text\" ref=\"inpt\"> \u003Cbr>\u003Cbr>\n    \u003Cbutton @click=\"showLog\">点我打印内容\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript lang=\"ts\" setup name=\"Person\">\n  import {ref} from 'vue'\n\t\n  let title1 = ref()\n  let title2 = ref()\n  let title3 = ref()\n\n  function showLog(){\n    \u002F\u002F 通过id获取元素\n    const t1 = document.getElementById('title1')\n    \u002F\u002F 打印内容\n    console.log((t1 as HTMLElement).innerText)\n    console.log((\u003CHTMLElement>t1).innerText)\n    console.log(t1?.innerText)\n    \n\t\t\u002F************************************\u002F\n\t\t\n    \u002F\u002F 通过ref获取元素\n    console.log(title1.value)\n    console.log(title2.value)\n    console.log(title3.value)\n  }\n\u003C\u002Fscript>\n```\n\n用在组件标签上：\n\n```vue\n\u003C!-- 父组件App.vue -->\n\u003Ctemplate>\n  \u003CPerson ref=\"ren\"\u002F>\n  \u003Cbutton @click=\"test\">测试\u003C\u002Fbutton>\n\u003C\u002Ftemplate>\n\n\u003Cscript lang=\"ts\" setup name=\"App\">\n  import Person from '.\u002Fcomponents\u002FPerson.vue'\n  import {ref} from 'vue'\n\n  let ren = ref()\n\n  function test(){\n    console.log(ren.value.name)\n    console.log(ren.value.age)\n  }\n\u003C\u002Fscript>\n\n\n\u003C!-- 子组件Person.vue中要使用defineExpose暴露内容 -->\n\u003Cscript lang=\"ts\" setup name=\"Person\">\n  import {ref,defineExpose} from 'vue'\n\t\u002F\u002F 数据\n  let name = ref('张三')\n  let age = ref(18)\n  \u002F****************************\u002F\n  \u002F****************************\u002F\n  \u002F\u002F 使用defineExpose将组件中的数据交给外部\n  defineExpose({name,age})\n\u003C\u002Fscript>\n```\n\n\n\n## 3.12. 【props】\n\n> ```js\n>\u002F\u002F 定义一个接口，限制每个Person对象的格式\n> export interface PersonInter {\n>  id:string,\n>  name:string,\n>     age:number\n>    }\n>    \n> \u002F\u002F 定义一个自定义类型Persons\n> export type Persons = Array\u003CPersonInter>\n> ```\n> \n> `App.vue`中代码：\n>\n> ```vue\n>\u003Ctemplate>\n> \t\u003CPerson :list=\"persons\"\u002F>\n> \u003C\u002Ftemplate>\n>   \n> \u003Cscript lang=\"ts\" setup name=\"App\">\n>   import Person from '.\u002Fcomponents\u002FPerson.vue'\n>   import {reactive} from 'vue'\n>     import {type Persons} from '.\u002Ftypes'\n>   \n>     let persons = reactive\u003CPersons>([\n>      {id:'e98219e12',name:'张三',age:18},\n>       {id:'e98219e13',name:'李四',age:19},\n>        {id:'e98219e14',name:'王五',age:20}\n>      ])\n>    \u003C\u002Fscript>\n>   \n> ```\n> \n> `Person.vue`中代码：\n>\n> ```Vue\n>\u003Ctemplate>\n> \u003Cdiv class=\"person\">\n>  \u003Cul>\n>      \u003Cli v-for=\"item in list\" :key=\"item.id\">\n>         {{item.name}}--{{item.age}}\n>       \u003C\u002Fli>\n>     \u003C\u002Ful>\n>    \u003C\u002Fdiv>\n>    \u003C\u002Ftemplate>\n>   \n> \u003Cscript lang=\"ts\" setup name=\"Person\">\n> import {defineProps} from 'vue'\n> import {type PersonInter} from '@\u002Ftypes'\n>   \n>   \u002F\u002F 第一种写法：仅接收\n> \u002F\u002F const props = defineProps(['list'])\n>   \n>   \u002F\u002F 第二种写法：接收+限制类型\n> \u002F\u002F defineProps\u003C{list:Persons}>()\n>   \n>   \u002F\u002F 第三种写法：接收+限制类型+指定默认值+限制必要性\n> let props = withDefaults(defineProps\u003C{list?:Persons}>(),{\n>      list:()=>[{id:'asdasg01',name:'小猪佩奇',age:18}]\n>   })\n>    console.log(props)\n>   \u003C\u002Fscript>\n>   ```\n> \n\n## 3.13. 【生命周期】\n\n* 概念：`Vue`组件实例在创建时要经历一系列的初始化步骤，在此过程中`Vue`会在合适的时机，调用特定的函数，从而让开发者有机会在特定阶段运行自己的代码，这些特定的函数统称为：生命周期钩子\n\n* 规律：\n\n  > 生命周期整体分为四个阶段，分别是：**创建、挂载、更新、销毁**，每个阶段都有两个钩子，一前一后。\n\n* `Vue2`的生命周期\n\n  > 创建阶段：`beforeCreate`、`created`\n  >\n  > 挂载阶段：`beforeMount`、`mounted`\n  >\n  > 更新阶段：`beforeUpdate`、`updated`\n  >\n  > 销毁阶段：`beforeDestroy`、`destroyed`\n\n* `Vue3`的生命周期\n\n  > 创建阶段：`setup`\n  >\n  > 挂载阶段：`onBeforeMount`、`onMounted`\n  >\n  > 更新阶段：`onBeforeUpdate`、`onUpdated`\n  >\n  > 卸载阶段：`onBeforeUnmount`、`onUnmounted`\n\n* 常用的钩子：`onMounted`(挂载完毕)、`onUpdated`(更新完毕)、`onBeforeUnmount`(卸载之前)\n\n* 示例代码：\n\n  ```vue\n  \u003Ctemplate>\n    \u003Cdiv class=\"person\">\n      \u003Ch2>当前求和为：{{ sum }}\u003C\u002Fh2>\n      \u003Cbutton @click=\"changeSum\">点我sum+1\u003C\u002Fbutton>\n    \u003C\u002Fdiv>\n  \u003C\u002Ftemplate>\n  \n  \u003C!-- vue3写法 -->\n  \u003Cscript lang=\"ts\" setup name=\"Person\">\n    import { \n      ref, \n      onBeforeMount, \n      onMounted, \n      onBeforeUpdate, \n      onUpdated, \n      onBeforeUnmount, \n      onUnmounted \n    } from 'vue'\n  \n    \u002F\u002F 数据\n    let sum = ref(0)\n    \u002F\u002F 方法\n    function changeSum() {\n      sum.value += 1\n    }\n    console.log('setup')\n    \u002F\u002F 生命周期钩子\n    onBeforeMount(()=>{\n      console.log('挂载之前')\n    })\n    onMounted(()=>{\n      console.log('挂载完毕')\n    })\n    onBeforeUpdate(()=>{\n      console.log('更新之前')\n    })\n    onUpdated(()=>{\n      console.log('更新完毕')\n    })\n    onBeforeUnmount(()=>{\n      console.log('卸载之前')\n    })\n    onUnmounted(()=>{\n      console.log('卸载完毕')\n    })\n  \u003C\u002Fscript>\n  ```\n\n## 3.14. 【自定义hook】\n\n- 什么是`hook`？—— 本质是一个函数，把`setup`函数中使用的`Composition API`进行了封装，类似于`vue2.x`中的`mixin`。\n\n- 自定义`hook`的优势：复用代码, 让`setup`中的逻辑更清楚易懂。\n\n示例代码：\n\n- `useSum.ts`中内容如下：\n\n  ```js\n  import {ref,onMounted} from 'vue'\n  \n  export default function(){\n    let sum = ref(0)\n  \n    const increment = ()=>{\n      sum.value += 1\n    }\n    const decrement = ()=>{\n      sum.value -= 1\n    }\n    onMounted(()=>{\n      increment()\n    })\n  \n    \u002F\u002F向外部暴露数据\n    return {sum,increment,decrement}\n  }\t\t\n  ```\n  \n- `useDog.ts`中内容如下：\n\n  ```js\n  import {reactive,onMounted} from 'vue'\n  import axios,{AxiosError} from 'axios'\n  \n  export default function(){\n    let dogList = reactive\u003Cstring[]>([])\n  \n    \u002F\u002F 方法\n    async function getDog(){\n      try {\n        \u002F\u002F 发请求\n        let {data} = await axios.get('https:\u002F\u002Fdog.ceo\u002Fapi\u002Fbreed\u002Fpembroke\u002Fimages\u002Frandom')\n        \u002F\u002F 维护数据\n        dogList.push(data.message)\n      } catch (error) {\n        \u002F\u002F 处理错误\n        const err = \u003CAxiosError>error\n        console.log(err.message)\n      }\n    }\n  \n    \u002F\u002F 挂载钩子\n    onMounted(()=>{\n      getDog()\n    })\n  \t\n    \u002F\u002F向外部暴露数据\n    return {dogList,getDog}\n  }\n  ```\n\n- 组件中具体使用：\n\n  ```vue\n  \u003Ctemplate>\n    \u003Ch2>当前求和为：{{sum}}\u003C\u002Fh2>\n    \u003Cbutton @click=\"increment\">点我+1\u003C\u002Fbutton>\n    \u003Cbutton @click=\"decrement\">点我-1\u003C\u002Fbutton>\n    \u003Chr>\n    \u003Cimg v-for=\"(u,index) in dogList.urlList\" :key=\"index\" :src=\"(u as string)\"> \n    \u003Cspan v-show=\"dogList.isLoading\">加载中......\u003C\u002Fspan>\u003Cbr>\n    \u003Cbutton @click=\"getDog\">再来一只狗\u003C\u002Fbutton>\n  \u003C\u002Ftemplate>\n  \n  \u003Cscript lang=\"ts\">\n    import {defineComponent} from 'vue'\n  \n    export default defineComponent({\n      name:'App',\n    })\n  \u003C\u002Fscript>\n  \n  \u003Cscript setup lang=\"ts\">\n    import useSum from '.\u002Fhooks\u002FuseSum'\n    import useDog from '.\u002Fhooks\u002FuseDog'\n  \t\n    let {sum,increment,decrement} = useSum()\n    let {dogList,getDog} = useDog()\n  \u003C\u002Fscript>\n  ```\n\n    \n\n---\n\n# 4. 路由\n\n## 4.1. 【对路由的理解】\n\nVue Router 是 Vue 官方的客户端路由解决方案。\n\n客户端路由的作用是在单页应用 (SPA) 中将浏览器的 URL 和用户看到的内容绑定起来。当用户在应用中浏览不同页面时，URL 会随之更新，但页面不需要从服务器重新加载。 \n\n## 4.2. 【基本切换效果】\n\n- `Vue3`中要使用`vue-router`的最新版本，目前是`4`版本。\n\n- 安装Vue Router.   npm i vue-router\n\n- 路由配置文件代码如下：\n\n- 创建新的文件夹router > index.ts\n\n  ```js\n  \u002F\u002F 创建一个路由器，并暴露出去\n  \n  \u002F\u002F 第一步：引入creatRouter\n  import {createRouter,createWebHistory} from 'vue-router'\n  \n  \u002F\u002F 引入一个一个可能要呈现的组件\n  import Home from '@\u002Fpages\u002FHome.vue'\n  import News from '@\u002Fpages\u002FNews.vue'\n  import About from '@\u002Fpages\u002FAbout.vue'\n  \n  \u002F\u002F 第二步：创建路由器\n  const router = createRouter({\n  \thistory:createWebHistory(),    \u002F\u002F 路由的工作模式\n  \troutes:[     \u002F\u002F 一个一个的路由规则\n  \t\t{\n  \t\t\tpath:'\u002Fhome',\n  \t\t\tcomponent:Home\n  \t\t},\n  \t\t{\n  \t\t\tpath:'\u002Fabout',\n  \t\t\tcomponent:About\n  \t\t}\n  \t]\n  })\n  \u002F\u002F 第三步：暴露出去router\n  export default router\n  ```\n* `main.ts`代码如下：\n\n  ```js\n  \u002F\u002F 引入 creatApp 用于创建应用\n  import { createApp } from \"vue\";\n  \n  \u002F\u002F 引入App根组件\n  import App from \".\u002FApp.vue\";\n  \u002F\u002F 引入路由器\n  import router from '.\u002Frouter\u002Findex'\n  \n  \u002F\u002F 创建一个应用\n   const app = createApp(App)\n  \u002F\u002F  使用路由器\n  app.use(router)\n  \u002F\u002F 挂载到整个应用到app容器中\n  app.mount('#app')\n  ```\n\n- `App.vue`代码如下\n\n  ```vue\n  \u003Ctemplate>\n    \u003Cdiv class=\"app\">\n      \u003Ch2 class=\"title\">Vue路由测试\u003C\u002Fh2>\n      \u003C!-- 导航区 -->\n      \u003Cdiv class=\"navigate\">\n        \u003CRouterLink to=\"\u002Fhome\" active-class=\"active\">首页\u003C\u002FRouterLink>\n        \u003CRouterLink to=\"\u002Fnews\" active-class=\"active\">新闻\u003C\u002FRouterLink>\n        \u003CRouterLink to=\"\u002Fabout\" active-class=\"active\">关于\u003C\u002FRouterLink>\n      \u003C\u002Fdiv>\n      \u003C!-- 展示区 -->\n      \u003Cdiv class=\"main-content\">\n        \u003CRouterView>\u003C\u002FRouterView>\n      \u003C\u002Fdiv>\n    \u003C\u002Fdiv>\n  \u003C\u002Ftemplate>\n  \n  \u003Cscript lang=\"ts\" setup name=\"App\">\n    import {RouterLink,RouterView} from 'vue-router'  \n      \n      \u002F\u002F不同于常规的 \u003Ca> 标签，我们使用组件 RouterLink 来创建链接。这使得 Vue Router 能够在不重新加载页面的情况下改变 URL，处理 URL 的生成、编码和其他功能。我们将会在之后的部分深入了解 RouterLink 组件。\n      \n      \u002F\u002FRouterView 组件可以使 Vue Router 知道你想要在哪里渲染当前 URL 路径对应的路由组件。它不一定要在 App.vue 中，你可以把它放在任何地方，但它需要在某处被导入，否则 Vue Router 就不会渲染任何东西。\n  \u003C\u002Fscript>\n  \n  ```\n\n## 4.3. 【两个注意点】\n\n> 1. 路由组件通常存放在`pages` 或 `views`文件夹，一般组件通常存放在`components`文件夹。\n>\n> 2. 通过点击导航，视觉效果上“消失” 了的路由组件，默认是被**卸载**掉的，需要的时候再去**挂载**。\n\n## 4.4.【路由器工作模式】\n\n1. `history`模式\n\n   > 优点：`URL`更加美观，不带有`#`，更接近传统的网站`URL`。\n   >\n   > 缺点：后期项目上线，需要服务端配合处理路径问题，否则刷新会有`404`错误。\n   >\n   > ```js\n   > const router = createRouter({\n   >   \thistory:createWebHistory(), \u002F\u002Fhistory模式\n   >   \t\u002F******\u002F\n   > })\n   > ```\n\n2. `hash`模式\n\n   > 优点：兼容性更好，因为不需要服务器端处理路径。\n   >\n   > 缺点：`URL`带有`#`不太美观，且在`SEO`优化方面相对较差。\n   >\n   > ```js\n   > const router = createRouter({\n   >   \thistory:createWebHashHistory(), \u002F\u002Fhash模式\n   >   \t\u002F******\u002F\n   > })\n   > ```\n\n## 4.5. 【to的两种写法】\n\n```vue\n\u003C!-- 第一种：to的字符串写法 -->\n\u003Crouter-link active-class=\"active\" to=\"\u002Fhome\">主页\u003C\u002Frouter-link>\n\n\u003C!-- 第二种：to的对象写法 -->\n\u003Crouter-link active-class=\"active\" :to=\"{path:'\u002Fhome'}\">Home\u003C\u002Frouter-link>\n```\n\n## 4.6. 【命名路由】\n\n作用：可以简化路由跳转及传参（后面就讲）。\n\n给路由规则命名：\n\n```js\nroutes:[\n  {\n    name:'zhuye',\n    path:'\u002Fhome',\n    component:Home\n  },\n  {\n    name:'xinwen',\n    path:'\u002Fnews',\n    component:News,\n  },\n  {\n    name:'guanyu',\n    path:'\u002Fabout',\n    component:About\n  }\n]\n```\n\n跳转路由：\n\n```vue\n\u003C!--简化前：需要写完整的路径（to的字符串写法） -->\n\u003Crouter-link to=\"\u002Fnews\u002Fdetail\">跳转\u003C\u002Frouter-link>\n\n\u003C!--简化后：直接通过名字跳转（to的对象写法配合name属性） -->\n\u003Crouter-link :to=\"{name:'guanyu'}\">跳转\u003C\u002Frouter-link>\n```\n\n\n\n## 4.7. 【嵌套路由】\n\n1. 编写`News`的子路由：`Detail.vue`\n\n2. 配置路由规则，使用`children`配置项：\n\n   ```ts\n   const router = createRouter({\n     history:createWebHistory(),\n   \troutes:[\n   \t\t{\n   \t\t\tname:'zhuye',\n   \t\t\tpath:'\u002Fhome',\n   \t\t\tcomponent:Home\n   \t\t},\n   \t\t{\n   \t\t\tname:'xinwen',\n   \t\t\tpath:'\u002Fnews',\n   \t\t\tcomponent:News,\n   \t\t\tchildren:[\n   \t\t\t\t{\n   \t\t\t\t\tname:'xiang',\n   \t\t\t\t\tpath:'detail',\n   \t\t\t\t\tcomponent:Detail\n   \t\t\t\t}\n   \t\t\t]\n   \t\t},\n   \t\t{\n   \t\t\tname:'guanyu',\n   \t\t\tpath:'\u002Fabout',\n   \t\t\tcomponent:About\n   \t\t}\n   \t]\n   })\n   export default router\n   ```\n   \n3. 跳转路由（记得要加完整路径）：\n\n   ```vue\n   \u003Crouter-link to=\"\u002Fnews\u002Fdetail\">xxxx\u003C\u002Frouter-link>\n   \u003C!-- 或 -->\n   \u003Crouter-link :to=\"{path:'\u002Fnews\u002Fdetail'}\">xxxx\u003C\u002Frouter-link>\n   ```\n\n4. 记得去`Home`组件中预留一个`\u003Crouter-view>`\n\n   ```vue\n   \u003Ctemplate>\n     \u003Cdiv class=\"news\">\n       \u003Cnav class=\"news-list\">\n         \u003CRouterLink v-for=\"news in newsList\" :key=\"news.id\" :to=\"{path:'\u002Fnews\u002Fdetail'}\">\n           {{news.name}}\n         \u003C\u002FRouterLink>\n       \u003C\u002Fnav>\n       \u003Cdiv class=\"news-detail\">\n         \u003CRouterView\u002F>\n       \u003C\u002Fdiv>\n     \u003C\u002Fdiv>\n   \u003C\u002Ftemplate>\n   ```\n\n   \n\n## 4.8. 【路由传参】\n\n### query参数\n\n   1. 传递参数\n\n      ```vue\n      \u003C!-- 跳转并携带query参数（to的字符串写法） -->\n      \u003Crouter-link to=\"\u002Fnews\u002Fdetail?a=1&b=2&content=欢迎你\">\n      \t跳转\n      \u003C\u002Frouter-link>\n      \t\t\t\t\n      \u003C!-- 跳转并携带query参数（to的对象写法） -->\n      \u003CRouterLink \n        :to=\"{\n          \u002F\u002Fname:'xiang', \u002F\u002F用name也可以跳转\n          path:'\u002Fnews\u002Fdetail',\n          query:{\n            id:news.id,\n            title:news.title,\n            content:news.content\n          }\n        }\"\n      >\n        {{news.title}}\n      \u003C\u002FRouterLink>\n      ```\n\n   2. 接收参数：\n\n      ```js\n      import {useRoute} from 'vue-router'\n      const route = useRoute()\n      \u002F\u002F 打印query参数\n      console.log(route.query)\n      ```\n\n\n### params参数\n\n   1. 传递参数\n\n      ```vue\n      \u003C!-- 跳转并携带params参数（to的字符串写法） -->\n      \u003CRouterLink :to=\"`\u002Fnews\u002Fdetail\u002F001\u002F新闻001\u002F内容001`\">{{news.title}}\u003C\u002FRouterLink>\n      \t\t\t\t\n      \u003C!-- 跳转并携带params参数（to的对象写法） -->\n      \u003CRouterLink \n        :to=\"{\n          name:'xiang', \u002F\u002F用name跳转\n          params:{\n            id:news.id,\n            title:news.title,\n            content:news.title\n          }\n        }\"\n      >\n        {{news.title}}\n      \u003C\u002FRouterLink>\n      ```\n\n   2. 接收参数：\n\n      ```js\n      import {useRoute} from 'vue-router'\n      const route = useRoute()\n      \u002F\u002F 打印params参数\n      console.log(route.params)\n      ```\n\n> 备注1：传递`params`参数时，若使用`to`的对象写法，必须使用`name`配置项，不能用`path`。\n>\n> 备注2：传递`params`参数时，需要提前在规则中占位。\n\n## 4.9. 【路由的props配置】\n\n作用：让路由组件更方便的收到参数（可以将路由参数作为`props`传给组件）\n\n```js\n{\n\tname:'xiang',\n\tpath:'detail\u002F:id\u002F:title\u002F:content',\n\tcomponent:Detail,\n\n  \u002F\u002F props的对象写法，作用：把对象中的每一组key-value作为props传给Detail组件\n  \u002F\u002F props:{a:1,b:2,c:3}, \n\n  \u002F\u002F props的布尔值写法，作用：把收到了每一组params参数，作为props传给Detail组件\n  \u002F\u002F props:true\n  \n  \u002F\u002F props的函数写法，作用：把返回的对象中每一组key-value作为props传给Detail组件\n  props(route){\n    return route.query\n  }\n}\n```\n\n## 4.10. 【 replace属性】\n\n  1. 作用：控制路由跳转时操作浏览器历史记录的模式。\n\n  2. 浏览器的历史记录有两种写入方式：分别为```push```和```replace```：\n\n     - ```push```是追加历史记录（默认值）。\n     - `replace`是替换当前记录。\n\n  3. 开启`replace`模式：\n\n     ```vue\n     \u003CRouterLink replace .......>News\u003C\u002FRouterLink>\n     ```\n\n## 4.11. 【编程式导航】\n\n路由组件的两个重要的属性：`$route`和`$router`变成了两个`hooks`\n\n```js\nimport {useRoute,useRouter} from 'vue-router'\n\nconst route = useRoute()\nconst router = useRouter()\n\nconsole.log(route.query)\nconsole.log(route.parmas)\nconsole.log(router.push)\nconsole.log(router.replace)\n\n\n\u002F\u002F 倒计时跳转\nonMounted(() =>{\n        setTimeout(() =>{\n            router.push('\u002Fnews')\n            console.log('跳转成功');\n        },3000)\n    })\n\n\u002F\u002F 按钮点击跳转\nfunction showNews(news:NewInter){\n        \u002F\u002F 和to的用法相同，不过得传参数\n        router.push({\n            name:'xiangqing',\n                    query:{\n                        id:news.id,\n                        title:news.title,\n                        content:news.content\n                    }\n        })\n    }\n```\n\n## 4.12. 【重定向】\n\n1. 作用：将特定的路径，重新定向到已有路由。\n\n2. 具体编码：\n\n   ```js\n   {\n       path:'\u002F',\n       redirect:'\u002Fabout'\n   }\n   ```\n\n\n\n# 5. pinia \n\n[**Pinia是一个轻量级的状态管理库，专为Vue.js设计**。它允许在组件之间共享和管理状态，类似于Vuex，但提供了更简洁的API和更好的TypeScript集成。Pinia的优势在于其易用性和可扩展性，适合Vue 3的组合式API思维。](https:\u002F\u002Fblog.csdn.net\u002Ftyxjolin\u002Farticle\u002Fdetails\u002F130032199)\n\n## 5.1【准备一个效果】\n\n## ![image-20260115171150626](images\u002Fimage-20260115171150626.png)\n\n了解了一个id库，**npm i nanoid**\n\n```javascript\n\u002F\u002F 引入id字符串随机生成库\n    import { nanoid } from 'nanoid';\n\n\u002F\u002F nanoid是一个函数直接用\n\u002F\u002F 把请求回来的情话，包装成一个对象，插入到talkList中\n        let obj = {id:nanoid(),title:data}\n```\n\n\n\n## 5.2【搭建 pinia 环境】\n\n第一步：`npm install pinia`\n\n第二步：操作`src\u002Fmain.ts`\n\n```ts\nimport { createApp } from 'vue'\nimport App from '.\u002FApp.vue'\n\n\u002F* 引入createPinia，用于创建pinia *\u002F\nimport { createPinia } from 'pinia'\n\n\u002F* 创建pinia *\u002F\nconst pinia = createPinia()\nconst app = createApp(App)\n\n\u002F* 使用插件 *\u002F{}\napp.use(pinia)\napp.mount('#app')\n```\n\n此时开发者工具中已经有了`pinia`选项\n\n\u003Cimg src=\"https:\u002F\u002Fcdn.nlark.com\u002Fyuque\u002F0\u002F2023\u002Fpng\u002F35780599\u002F1684309952481-c67f67f9-d1a3-4d69-8bd6-2b381e003f31.png\" style=\"zoom:80%;border:1px solid black;border-radius:10px\" \u002F>\n\n## 5.3【存储+读取数据】\n\n1. `Store`是一个保存：**状态**、**业务逻辑** 的实体，每个组件都可以**读取**、**写入**它。\n\n2. 它有三个概念：`state`、`getter`、`action`，相当于组件中的： `data`、 `computed` 和 `methods`。\n\n3. 具体编码：`src\u002Fstore\u002Fcount.ts`\n\n   ```ts\n   \u002F\u002F 引入defineStore用于创建store\n   import {defineStore} from 'pinia'\n   \n   \u002F\u002F 定义并暴露一个store\n   export const useCountStore = defineStore('count',{\n     \u002F\u002F 动作\n     actions:{},\n     \u002F\u002F 状态\n     state(){\n       return {\n         sum:6\n       }\n     },\n     \u002F\u002F 计算\n     getters:{}\n   })\n   ```\n\n4. 具体编码：`src\u002Fstore\u002Ftalk.ts`\n\n   ```js\n   \u002F\u002F 引入defineStore用于创建store\n   import {defineStore} from 'pinia'\n   \n   \u002F\u002F 定义并暴露一个store\n   export const useTalkStore = defineStore('talk',{\n     \u002F\u002F 动作\n     actions:{},\n     \u002F\u002F 状态\n     state(){\n       return {\n         talkList:[\n           {id:'yuysada01',content:'你今天有点怪，哪里怪？怪好看的！'},\n        \t\t{id:'yuysada02',content:'草莓、蓝莓、蔓越莓，你想我了没？'},\n           {id:'yuysada03',content:'心里给你留了一块地，我的死心塌地'}\n         ]\n       }\n     },\n     \u002F\u002F 计算\n     getters:{}\n   })\n   ```\n   \n5. 组件中使用`state`中的数据\n\n   ```vue\n   \u003Ctemplate>\n     \u003Ch2>当前求和为：{{ sumStore.sum }}\u003C\u002Fh2>\n   \u003C\u002Ftemplate>\n   \n   \u003Cscript setup lang=\"ts\" name=\"Count\">\n     \u002F\u002F 引入对应的useXxxxxStore\t\n     import {useSumStore} from '@\u002Fstore\u002Fsum'\n     \n     \u002F\u002F 调用useXxxxxStore得到对应的store\n     const sumStore = useSumStore()\n   \u003C\u002Fscript>\n   ```\n\n   ```vue\n   \u003Ctemplate>\n   \t\u003Cul>\n       \u003Cli v-for=\"talk in talkStore.talkList\" :key=\"talk.id\">\n         {{ talk.content }}\n       \u003C\u002Fli>\n     \u003C\u002Ful>\n   \u003C\u002Ftemplate>\n   \n   \u003Cscript setup lang=\"ts\" name=\"Count\">\n     import axios from 'axios'\n     import {useTalkStore} from '@\u002Fstore\u002Ftalk'\n   \n     const talkStore = useTalkStore()\n   \u003C\u002Fscript>\n   ```\n\n   \n\n## 5.4.【修改数据】(三种方式)\n\n1. 第一种修改方式，直接修改\n\n   ```ts\n   countStore.sum = 666\n   ```\n\n2. 第二种修改方式：批量修改\n\n   ```ts\n   countStore.$patch({\n     sum:999,\n     school:'atguigu'\n   })\n   ```\n\n3. 第三种修改方式：借助`action`修改（`action`中可以编写一些业务逻辑）\n\n   ```js\n   import { defineStore } from 'pinia'\n   \n   export const useCountStore = defineStore('count', {\n     \u002F*************\u002F\n     actions: {\n       \u002F\u002F加\n       increment(value:number) {\n         if (this.sum \u003C 10) {\n           \u002F\u002F操作countStore中的sum\n           this.sum += value\n         }\n       },\n       \u002F\u002F减\n       decrement(value:number){\n         if(this.sum > 1){\n           this.sum -= value\n         }\n       }\n     },\n     \u002F*************\u002F\n   })\n   ```\n\n4. 组件中调用`action`即可\n\n   ```js\n   \u002F\u002F 使用countStore\n   const countStore = useCountStore()\n   \n   \u002F\u002F 调用对应action\n   countStore.incrementOdd(n.value)\n   ```\n\n\n## 5.5.【storeToRefs】\n\n- 借助`storeToRefs`将`store`中的数据转为`ref`对象，方便在模板中使用。\n  - 注意：`pinia`提供的`storeToRefs`只会将数据做转换，而`Vue`的`toRefs`会转换`store`中的方法，浪费内存。\n\n```vue\n\u003Ctemplate>\n\t\u003Cdiv class=\"count\">\n\t\t\u003Ch2>当前求和为：{{sum}}\u003C\u002Fh2>\n\t\u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript setup lang=\"ts\" name=\"Count\">\n  import { useCountStore } from '@\u002Fstore\u002Fcount'\n  \u002F* 引入storeToRefs *\u002F\n  import { storeToRefs } from 'pinia'\n\n\t\u002F* 得到countStore *\u002F\n  const countStore = useCountStore()\n  \u002F* 使用storeToRefs转换countStore，随后解构 *\u002F\n  const {sum} = storeToRefs(countStore)\n\u003C\u002Fscript>\n\n```\n\n## 5.6.【getters】\n\n  1. 概念：当`state`中的数据，需要经过处理后再使用时，可以使用`getters`配置。\n\n  2. 追加```getters```配置。\n\n     ```js\n     \u002F\u002F 引入defineStore用于创建store\n     import {defineStore} from 'pinia'\n     \n     \u002F\u002F 定义并暴露一个store\n     export const useCountStore = defineStore('count',{\n       \u002F\u002F 动作\n       actions:{\n         \u002F************\u002F\n       },\n       \u002F\u002F 状态\n       state(){\n         return {\n           sum:1,\n           school:'atguigu'\n         }\n       },\n       \u002F\u002F 计算\n       getters:{\n         bigSum:(state):number => state.sum *10,\n         upperSchool():string{\n           return this. school.toUpperCase()\n         }\n       }\n     })\n     ```\n\n  3. 组件中读取数据：\n\n     ```js\n     const {increment,decrement} = countStore\n     let {sum,school,bigSum,upperSchool} = storeToRefs(countStore)\n     ```\n\n     \n\n## 5.7.【$subscribe】\n\n通过 store 的 `$subscribe()` 方法侦听 `state` 及其变化\n\n```ts\ntalkStore.$subscribe((mutate,state)=>{\n  console.log('LoveTalk',mutate,state)\n  localStorage.setItem('talk',JSON.stringify(talkList.value))\n})\n```\n\n\n\n## 5.8. 【store组合式写法】\n\n```ts\nimport {defineStore} from 'pinia'\nimport axios from 'axios'\nimport {nanoid} from 'nanoid'\nimport {reactive} from 'vue'\n\nexport const useTalkStore = defineStore('talk',()=>{\n  \u002F\u002F talkList就是state\n  const talkList = reactive(\n    JSON.parse(localStorage.getItem('talkList') as string) || []\n  )\n\n  \u002F\u002F getATalk函数相当于action\n  async function getATalk(){\n    \u002F\u002F 发请求，下面这行的写法是：连续解构赋值+重命名\n    let {data:{content:title}} = await axios.get('https:\u002F\u002Fapi.uomg.com\u002Fapi\u002Frand.qinghua?format=json')\n    \u002F\u002F 把请求回来的字符串，包装成一个对象\n    let obj = {id:nanoid(),title}\n    \u002F\u002F 放到数组中\n    talkList.unshift(obj)\n  }\n  return {talkList,getATalk}\n})\n```\n\n\n\n# 6. 组件通信\n\n**`Vue3`组件通信和`Vue2`的区别：**\n\n* 移出事件总线，使用`mitt`代替。\n\n- `vuex`换成了`pinia`。\n- 把`.sync`优化到了`v-model`里面了。\n- 把`$listeners`所有的东西，合并到`$attrs`中了。\n- `$children`被砍掉了。\n\n**常见搭配形式：**\n\n\u003Cimg src=\"images\u002Fimage-20231119185900990.png\" alt=\"image-20231119185900990\" style=\"zoom:60%;\" \u002F> \n\n## 6.1. 【props】\n\n概述：`props`是使用频率最高的一种通信方式，常用与 ：**父 ↔ 子**。\n\n- 若 **父传子**：属性值是**非函数**。\n- 若 **子传父**：属性值是**函数**。\n\n父组件：\n\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"father\">\n    \u003Ch3>父组件，\u003C\u002Fh3>\n\t\t\u003Ch4>我的车：{{ car }}\u003C\u002Fh4>\n\t\t\u003Ch4>儿子给的玩具：{{ toy }}\u003C\u002Fh4>\n\t\t\u003CChild :car=\"car\" :getToy=\"getToy\"\u002F>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript setup lang=\"ts\" name=\"Father\">\n\timport Child from '.\u002FChild.vue'\n\timport { ref } from \"vue\";\n\t\u002F\u002F 数据\n\tconst car = ref('奔驰')\n\tconst toy = ref()\n\t\u002F\u002F 方法\n\tfunction getToy(value:string){\n\t\ttoy.value = value\n\t}\n\u003C\u002Fscript>\n```\n\n子组件\n\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"child\">\n    \u003Ch3>子组件\u003C\u002Fh3>\n\t\t\u003Ch4>我的玩具：{{ toy }}\u003C\u002Fh4>\n\t\t\u003Ch4>父给我的车：{{ car }}\u003C\u002Fh4>\n\t\t\u003Cbutton @click=\"getToy(toy)\">玩具给父亲\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript setup lang=\"ts\" name=\"Child\">\n\timport { ref } from \"vue\";\n\tconst toy = ref('奥特曼')\n\t\n\tdefineProps(['car','getToy'])\n\u003C\u002Fscript>\n```\n\n## 6.2. 【自定义事件】\n\n1. 概述：自定义事件常用于：**子 => 父。**\n2. 注意区分好：原生事件、自定义事件。\n\n- 原生事件：\n  - 事件名是特定的（`click`、`mosueenter`等等）\t\n  - 事件对象`$event`: 是包含事件相关信息的对象（`pageX`、`pageY`、`target`、`keyCode`）\n- 自定义事件：\n  - 事件名是任意名称\n  - \u003Cstrong style=\"color:red\">事件对象`$event`: 是调用`emit`时所提供的数据，可以是任意类型！！！\u003C\u002Fstrong >\n\n3. 示例：\n\n   ```html\n   \u003C!--在父组件中，给子组件绑定自定义事件：-->\n   \u003CChild @send-toy=\"toy = $event\"\u002F>\n   \n   \u003C!--注意区分原生事件与自定义事件中的$event-->\n   \u003Cbutton @click=\"toy = $event\">测试\u003C\u002Fbutton>\n   ```\n\n   ```js\n   \u002F\u002F子组件中，触发事件：\n   this.$emit('send-toy', 具体数据)\n   ```\n\n## 6.3. 【mitt】\n\n概述：与消息订阅与发布（`pubsub`）功能类似，可以实现任意组件间通信。\n\n安装`mitt`\n\n```shell\nnpm i mitt\n```\n\n新建文件：`src\\utils\\emitter.ts`\n\n```javascript\n\u002F\u002F 引入mitt \nimport mitt from \"mitt\";\n\n\u002F\u002F 创建emitter\nconst emitter = mitt()\n\n\u002F*\n  \u002F\u002F 绑定事件\n  emitter.on('abc',(value)=>{\n    console.log('abc事件被触发',value)\n  })\n  emitter.on('xyz',(value)=>{\n    console.log('xyz事件被触发',value)\n  })\n\n  setInterval(() => {\n    \u002F\u002F 触发事件\n    emitter.emit('abc',666)\n    emitter.emit('xyz',777)\n  }, 1000);\n\n  setTimeout(() => {\n    \u002F\u002F 清理事件\n    emitter.all.clear()\n  }, 3000); \n*\u002F\n\n\u002F\u002F 创建并暴露mitt\nexport default emitter\n```\n\n接收数据的组件中：绑定事件、同时在销毁前解绑事件：\n\n```typescript\nimport emitter from \"@\u002Futils\u002Femitter\";\nimport { onUnmounted } from \"vue\";\n\n\u002F\u002F 绑定事件\nemitter.on('send-toy',(value)=>{\n  console.log('send-toy事件被触发',value)\n})\n\nonUnmounted(()=>{\n  \u002F\u002F 解绑事件\n  emitter.off('send-toy')\n})\n```\n\n【第三步】：提供数据的组件，在合适的时候触发事件\n\n```javascript\nimport emitter from \"@\u002Futils\u002Femitter\";\n\nfunction sendToy(){\n  \u002F\u002F 触发事件\n  emitter.emit('send-toy',toy.value)\n}\n```\n\n**注意这个重要的内置关系，总线依赖着这个内置关系**\n\n## 6.4.【v-model】\n\n1. 概述：实现 **父↔子** 之间相互通信。\n\n2. 前序知识 —— `v-model`的本质\n\n   ```vue\n   \u003C!-- 使用v-model指令 -->\n   \u003Cinput type=\"text\" v-model=\"userName\">\n   \n   \u003C!-- v-model的本质是下面这行代码 -->\n   \u003Cinput \n     type=\"text\" \n     :value=\"userName\" \n     @input=\"userName =(\u003CHTMLInputElement>$event.target).value\"\n   >\n   ```\n\n3. 组件标签上的`v-model`的本质：`:moldeValue` ＋ `update:modelValue`事件。\n\n   ```vue\n   \u003C!-- 组件标签上使用v-model指令 -->\n   \u003CAtguiguInput v-model=\"userName\"\u002F>\n   \n   \u003C!-- 组件标签上v-model的本质 -->\n   \u003CAtguiguInput :modelValue=\"userName\" @update:model-value=\"userName = $event\"\u002F>\n   ```\n\n   `AtguiguInput`组件中：\n\n   ```vue\n   \u003Ctemplate>\n     \u003Cdiv class=\"box\">\n       \u003C!--将接收的value值赋给input元素的value属性，目的是：为了呈现数据 -->\n   \t\t\u003C!--给input元素绑定原生input事件，触发input事件时，进而触发update:model-value事件-->\n       \u003Cinput \n          type=\"text\" \n          :value=\"modelValue\" \n          @input=\"emit('update:model-value',$event.target.value)\"\n       >\n     \u003C\u002Fdiv>\n   \u003C\u002Ftemplate>\n   \n   \u003Cscript setup lang=\"ts\" name=\"AtguiguInput\">\n     \u002F\u002F 接收props\n     defineProps(['modelValue'])\n     \u002F\u002F 声明事件\n     const emit = defineEmits(['update:model-value'])\n   \u003C\u002Fscript>\n   ```\n\n4. 也可以更换`value`，例如改成`abc`\n\n   ```vue\n   \u003C!-- 也可以更换value，例如改成abc-->\n   \u003CAtguiguInput v-model:abc=\"userName\"\u002F>\n   \n   \u003C!-- 上面代码的本质如下 -->\n   \u003CAtguiguInput :abc=\"userName\" @update:abc=\"userName = $event\"\u002F>\n   ```\n\n   `AtguiguInput`组件中：\n\n   ```vue\n   \u003Ctemplate>\n     \u003Cdiv class=\"box\">\n       \u003Cinput \n          type=\"text\" \n          :value=\"abc\" \n          @input=\"emit('update:abc',$event.target.value)\"\n       >\n     \u003C\u002Fdiv>\n   \u003C\u002Ftemplate>\n   \n   \u003Cscript setup lang=\"ts\" name=\"AtguiguInput\">\n     \u002F\u002F 接收props\n     defineProps(['abc'])\n     \u002F\u002F 声明事件\n     const emit = defineEmits(['update:abc'])\n   \u003C\u002Fscript>\n   ```\n\n5. 如果`value`可以更换，那么就可以在组件标签上多次使用`v-model`\n\n   ```vue\n   \u003CAtguiguInput v-model:abc=\"userName\" v-model:xyz=\"password\"\u002F>\n   ```\n\n   \n\n\n## 6.5.【$attrs 】\n\n1. 概述：`$attrs`用于实现**当前组件的父组件**，向**当前组件的子组件**通信（**祖→孙**）。\n\n2. 具体说明：`$attrs`是一个对象，包含所有父组件传入的标签属性。\n\n   >  注意：`$attrs`会自动排除`props`中声明的属性(可以认为声明过的 `props` 被子组件自己“消费”了)\n\n父组件：\n\n```vue\n\u003Ctemplate>\n  \u003Cdiv class=\"father\">\n    \u003Ch3>父组件\u003C\u002Fh3>\n\t\t\u003CChild :a=\"a\" :b=\"b\" :c=\"c\" :d=\"d\" v-bind=\"{x:100,y:200}\" :updateA=\"updateA\"\u002F>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript setup lang=\"ts\" name=\"Father\">\n\timport Child from '.\u002FChild.vue'\n\timport { ref } from \"vue\";\n\tlet a = ref(1)\n\tlet b = ref(2)\n\tlet c = ref(3)\n\tlet d = ref(4)\n\n\tfunction updateA(value){\n\t\ta.value = value\n\t}\n\u003C\u002Fscript>\n```\n\n子组件：\n\n```vue\n\u003Ctemplate>\n\t\u003Cdiv class=\"child\">\n\t\t\u003Ch3>子组件\u003C\u002Fh3>\n\t\t\u003CGrandChild v-bind=\"$attrs\"\u002F>\n\t\u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript setup lang=\"ts\" name=\"Child\">\n\timport GrandChild from '.\u002FGrandChild.vue'\n\u003C\u002Fscript>\n```\n\n孙组件：\n\n```vue\n\u003Ctemplate>\n\t\u003Cdiv class=\"grand-child\">\n\t\t\u003Ch3>孙组件\u003C\u002Fh3>\n\t\t\u003Ch4>a：{{ a }}\u003C\u002Fh4>\n\t\t\u003Ch4>b：{{ b }}\u003C\u002Fh4>\n\t\t\u003Ch4>c：{{ c }}\u003C\u002Fh4>\n\t\t\u003Ch4>d：{{ d }}\u003C\u002Fh4>\n\t\t\u003Ch4>x：{{ x }}\u003C\u002Fh4>\n\t\t\u003Ch4>y：{{ y }}\u003C\u002Fh4>\n\t\t\u003Cbutton @click=\"updateA(666)\">点我更新A\u003C\u002Fbutton>\n\t\u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript setup lang=\"ts\" name=\"GrandChild\">\n\tdefineProps(['a','b','c','d','x','y','updateA'])\n\u003C\u002Fscript>\n```\n\n## 6.6. 【$refs、$parent】\n\n1. 概述：\n\n   * `$refs`用于 ：**父→子。**\n   * `$parent`用于：**子→父。**\n\n2. 原理如下：\n\n   | 属性      | 说明                                                     |\n   | --------- | -------------------------------------------------------- |\n   | `$refs`   | 值为对象，包含所有被`ref`属性标识的`DOM`元素或组件实例。 |\n   | `$parent` | 值为对象，当前组件的父组件实例对象。                     |\n\n## 6.7. 【provide、inject】\n\n1. 概述：实现**祖孙组件**直接通信\n\n2. 具体使用：\n\n   * 在祖先组件中通过`provide`配置向后代组件提供数据\n   * 在后代组件中通过`inject`配置来声明接收数据\n\n4. 具体编码：\n\n   【第一步】父组件中，使用`provide`提供数据\n\n   ```vue\n   \u003Ctemplate>\n     \u003Cdiv class=\"father\">\n       \u003Ch3>父组件\u003C\u002Fh3>\n       \u003Ch4>资产：{{ money }}\u003C\u002Fh4>\n       \u003Ch4>汽车：{{ car }}\u003C\u002Fh4>\n       \u003Cbutton @click=\"money += 1\">资产+1\u003C\u002Fbutton>\n       \u003Cbutton @click=\"car.price += 1\">汽车价格+1\u003C\u002Fbutton>\n       \u003CChild\u002F>\n     \u003C\u002Fdiv>\n   \u003C\u002Ftemplate>\n   \n   \u003Cscript setup lang=\"ts\" name=\"Father\">\n     import Child from '.\u002FChild.vue'\n     import { ref,reactive,provide } from \"vue\";\n     \u002F\u002F 数据\n     let money = ref(100)\n     let car = reactive({\n       brand:'奔驰',\n       price:100\n     })\n     \u002F\u002F 用于更新money的方法\n     function updateMoney(value:number){\n       money.value += value\n     }\n     \u002F\u002F 提供数据\n     provide('moneyContext',{money,updateMoney})\n     provide('car',car)\n   \u003C\u002Fscript>\n   ```\n   \n   > 注意：子组件中不用编写任何东西，是不受到任何打扰的\n   \n   【第二步】孙组件中使用`inject`配置项接受数据。\n   \n   ```vue\n   \u003Ctemplate>\n     \u003Cdiv class=\"grand-child\">\n       \u003Ch3>我是孙组件\u003C\u002Fh3>\n       \u003Ch4>资产：{{ money }}\u003C\u002Fh4>\n       \u003Ch4>汽车：{{ car }}\u003C\u002Fh4>\n       \u003Cbutton @click=\"updateMoney(6)\">点我\u003C\u002Fbutton>\n     \u003C\u002Fdiv>\n   \u003C\u002Ftemplate>\n   \n   \u003Cscript setup lang=\"ts\" name=\"GrandChild\">\n     import { inject } from 'vue';\n     \u002F\u002F 注入数据\n    let {money,updateMoney} = inject('moneyContext',{money:0,updateMoney:(x:number)=>{}})\n     let car = inject('car')\n\u003C\u002Fscript>\n   ```\n\n\n## 6.8. 【pinia】\n\n参考之前`pinia`部分的讲解\n\n## 6.9. 【slot】\n\n### 1. 默认插槽\n\n![img](http:\u002F\u002F49.232.112.44\u002Fimages\u002Fdefault_slot.png)\n\n```vue\n父组件中：\n        \u003CCategory title=\"今日热门游戏\">\n          \u003Cul>\n            \u003Cli v-for=\"g in games\" :key=\"g.id\">{{ g.name }}\u003C\u002Fli>\n          \u003C\u002Ful>\n        \u003C\u002FCategory>\n子组件中：\n        \u003Ctemplate>\n          \u003Cdiv class=\"item\">\n            \u003Ch3>{{ title }}\u003C\u002Fh3>\n            \u003C!-- 默认插槽 -->\n            \u003Cslot>\u003C\u002Fslot>\n          \u003C\u002Fdiv>\n        \u003C\u002Ftemplate>\n```\n\n### 2. 具名插槽\n\n```vue\n父组件中：\n        \u003CCategory title=\"今日热门游戏\">\n          \u003Ctemplate v-slot:s1>\n            \u003Cul>\n              \u003Cli v-for=\"g in games\" :key=\"g.id\">{{ g.name }}\u003C\u002Fli>\n            \u003C\u002Ful>\n          \u003C\u002Ftemplate>\n          \u003Ctemplate #s2>\n            \u003Ca href=\"\">更多\u003C\u002Fa>\n          \u003C\u002Ftemplate>\n        \u003C\u002FCategory>\n子组件中：\n        \u003Ctemplate>\n          \u003Cdiv class=\"item\">\n            \u003Ch3>{{ title }}\u003C\u002Fh3>\n            \u003Cslot name=\"s1\">\u003C\u002Fslot>\n            \u003Cslot name=\"s2\">\u003C\u002Fslot>\n          \u003C\u002Fdiv>\n        \u003C\u002Ftemplate>\n```\n\n### 3. 作用域插槽 \n\n1. 理解：\u003Cspan style=\"color:red\">数据在组件的自身，但根据数据生成的结构需要组件的使用者来决定。\u003C\u002Fspan>（新闻数据在`News`组件中，但使用数据所遍历出来的结构由`App`组件决定）\n\n3. 具体编码：\n\n   ```vue\n   父组件中：\n         \u003CGame v-slot=\"params\">\n         \u003C!-- \u003CGame v-slot:default=\"params\"> -->\n         \u003C!-- \u003CGame #default=\"params\"> -->\n           \u003Cul>\n             \u003Cli v-for=\"g in params.games\" :key=\"g.id\">{{ g.name }}\u003C\u002Fli>\n           \u003C\u002Ful>\n         \u003C\u002FGame>\n   \n   子组件中：\n         \u003Ctemplate>\n           \u003Cdiv class=\"category\">\n             \u003Ch2>今日游戏榜单\u003C\u002Fh2>\n             \u003Cslot :games=\"games\" a=\"哈哈\">\u003C\u002Fslot>\n           \u003C\u002Fdiv>\n         \u003C\u002Ftemplate>\n   \n         \u003Cscript setup lang=\"ts\" name=\"Category\">\n           import {reactive} from 'vue'\n           let games = reactive([\n             {id:'asgdytsa01',name:'英雄联盟'},\n             {id:'asgdytsa02',name:'王者荣耀'},\n             {id:'asgdytsa03',name:'红色警戒'},\n             {id:'asgdytsa04',name:'斗罗大陆'}\n           ])\n         \u003C\u002Fscript>\n   ```\n\n\n\n# 7. 其它 API\n\n## 7.1.【shallowRef 与 shallowReactive 】\n\n### `shallowRef`\n\n1. 作用：创建一个响应式数据，但只对顶层属性进行响应式处理。\n\n2. 用法：\n\n   ```js\n   let myVar = shallowRef(initialValue);\n   ```\n\n3. 特点：只跟踪引用值的变化，不关心值内部的属性变化。\n\n### `shallowReactive`\n\n1. 作用：创建一个浅层响应式对象，只会使对象的最顶层属性变成响应式的，对象内部的嵌套属性则不会变成响应式的\n\n2. 用法：\n\n   ```js\n   const myObj = shallowReactive({ ... });\n   ```\n\n3. 特点：对象的顶层属性是响应式的，但嵌套对象的属性不是。\n\n### 总结\n\n> 通过使用 [`shallowRef()`](https:\u002F\u002Fcn.vuejs.org\u002Fapi\u002Freactivity-advanced.html#shallowref) 和 [`shallowReactive()`](https:\u002F\u002Fcn.vuejs.org\u002Fapi\u002Freactivity-advanced.html#shallowreactive) 来绕开深度响应。浅层式 `API` 创建的状态只在其顶层是响应式的，对所有深层的对象不会做任何处理，避免了对每一个内部属性做响应式所带来的性能成本，这使得属性的访问变得更快，可提升性能。\n\n\n\n## 7.2.【readonly 与 shallowReadonly】\n\n### **`readonly`**\n\n1. 作用：用于创建一个对象的深只读副本。\n\n2. 用法：\n\n   ```js\n   const original = reactive({ ... });\n   const readOnlyCopy = readonly(original);\n   ```\n\n3. 特点：\n\n   * 对象的所有嵌套属性都将变为只读。\n   * 任何尝试修改这个对象的操作都会被阻止（在开发模式下，还会在控制台中发出警告）。\n\n4. 应用场景：\n   * 创建不可变的状态快照。\n   * 保护全局状态或配置不被修改。\n\n### **`shallowReadonly`**\n\n1. 作用：与 `readonly` 类似，但只作用于对象的顶层属性。\n\n2. 用法：\n\n   ```js\n   const original = reactive({ ... });\n   const shallowReadOnlyCopy = shallowReadonly(original);\n   ```\n\n3. 特点：\n\n   * 只将对象的顶层属性设置为只读，对象内部的嵌套属性仍然是可变的。\n\n   * 适用于只需保护对象顶层属性的场景。\n\n     \n\n## 7.3.【toRaw 与 markRaw】\n\n### `toRaw`\n\n1. 作用：用于获取一个响应式对象的原始对象， `toRaw` 返回的对象不再是响应式的，不会触发视图更新。\n\n   > 官网描述：这是一个可以用于临时读取而不引起代理访问\u002F跟踪开销，或是写入而不触发更改的特殊方法。不建议保存对原始对象的持久引用，请谨慎使用。\n\n   > 何时使用？ —— 在需要将响应式对象传递给非 `Vue` 的库或外部系统时，使用 `toRaw` 可以确保它们收到的是普通对象\n\n2. 具体编码：\n\n   ```js\n   import { reactive,toRaw,markRaw,isReactive } from \"vue\";\n   \n   \u002F* toRaw *\u002F\n   \u002F\u002F 响应式对象\n   let person = reactive({name:'tony',age:18})\n   \u002F\u002F 原始对象\n   let rawPerson = toRaw(person)\n   \n   \n   \u002F* markRaw *\u002F\n   let citysd = markRaw([\n     {id:'asdda01',name:'北京'},\n     {id:'asdda02',name:'上海'},\n     {id:'asdda03',name:'天津'},\n     {id:'asdda04',name:'重庆'}\n   ])\n   \u002F\u002F 根据原始对象citys去创建响应式对象citys2 —— 创建失败，因为citys被markRaw标记了\n   let citys2 = reactive(citys)\n   console.log(isReactive(person))\n   console.log(isReactive(rawPerson))\n   console.log(isReactive(citys))\n   console.log(isReactive(citys2))\n   ```\n\n### `markRaw`\n\n1. 作用：标记一个对象，使其**永远不会**变成响应式的。\n\n   > 例如使用`mockjs`时，为了防止误把`mockjs`变为响应式对象，可以使用 `markRaw` 去标记`mockjs`\n\n2. 编码：\n\n   ```js\n   \u002F* markRaw *\u002F\n   let citys = markRaw([\n     {id:'asdda01',name:'北京'},\n     {id:'asdda02',name:'上海'},\n     {id:'asdda03',name:'天津'},\n     {id:'asdda04',name:'重庆'}\n   ])\n   \u002F\u002F 根据原始对象citys去创建响应式对象citys2 —— 创建失败，因为citys被markRaw标记了\n   let citys2 = reactive(citys)\n   ```\n\n## 7.4.【customRef】\n\n作用：创建一个自定义的`ref`，并对其依赖项跟踪和更新触发进行逻辑控制。\n\n实现防抖效果（`useSumRef.ts`）：\n\n```typescript\nimport {customRef } from \"vue\";\n\nexport default function(initValue:string,delay:number){\n  let msg = customRef((track,trigger)=>{\n    let timer:number\n    return {\n      get(){\n        track() \u002F\u002F 告诉Vue数据msg很重要，要对msg持续关注，一旦变化就更新\n        return initValue\n      },\n      set(value){\n        clearTimeout(timer)\n        timer = setTimeout(() => {\n          initValue = value\n          trigger() \u002F\u002F通知Vue数据msg变化了\n        }, delay);\n      }\n    }\n  }) \n  return {msg}\n}\n```\n\n组件中使用：\n\n\n\n\n\n# 8. Vue3新组件\n\n## 8.1. 【Teleport】\n\n- 什么是Teleport？—— Teleport 是一种能够将我们的**组件html结构**移动到指定位置的技术。\n\n```html\n\u003Cteleport to='body' >\n    \u003Cdiv class=\"modal\" v-show=\"isShow\">\n      \u003Ch2>我是一个弹窗\u003C\u002Fh2>\n      \u003Cp>我是弹窗中的一些内容\u003C\u002Fp>\n      \u003Cbutton @click=\"isShow = false\">关闭弹窗\u003C\u002Fbutton>\n    \u003C\u002Fdiv>\n\u003C\u002Fteleport>\n```\n\n## 8.2. 【Suspense】\n\n-  等待异步组件时渲染一些额外内容，让应用有更好的用户体验 \n-  使用步骤： \n   -  异步引入组件\n   -  使用`Suspense`包裹组件，并配置好`default` 与 `fallback`\n\n```tsx\nimport { defineAsyncComponent,Suspense } from \"vue\";\nconst Child = defineAsyncComponent(()=>import('.\u002FChild.vue'))\n```\n\n```vue\n\u003Ctemplate>\n    \u003Cdiv class=\"app\">\n        \u003Ch3>我是App组件\u003C\u002Fh3>\n        \u003CSuspense>\n          \u003Ctemplate v-slot:default>\n            \u003CChild\u002F>\n          \u003C\u002Ftemplate>\n          \u003Ctemplate v-slot:fallback>\n            \u003Ch3>加载中.......\u003C\u002Fh3>\n          \u003C\u002Ftemplate>\n        \u003C\u002FSuspense>\n    \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n```\n\n\n\n## 8.3.【全局API转移到应用对象】\n\n- `app.component`\n- `app.config`\n- `app.directive`\n- `app.mount`\n- `app.unmount`\n- `app.use`\n\n## 8.4.【其他】\n\n- 过渡类名 `v-enter` 修改为 `v-enter-from`、过渡类名 `v-leave` 修改为 `v-leave-from`。\n\n\n- `keyCode` 作为 `v-on` 修饰符的支持。\n\n- `v-model` 指令在组件上的使用已经被重新设计，替换掉了 `v-bind.sync。`\n\n- `v-if` 和 `v-for` 在同一个元素身上使用时的优先级发生了变化。\n\n- 移除了`$on`、`$off` 和 `$once` 实例方法。\n\n- 移除了过滤器 `filter`。\n\n- 移除了`$children` 实例 `propert`。\n\n  ......\n",null,"1",38,0,1,"2026-04-07T16:18:29.196Z","2026-04-07T16:18:29.200Z","2026-05-25T22:05:45.865Z","0",{"id":28,"categoryName":37,"slug":38,"description":39,"sort":30,"isEnable":31,"createTime":40,"updateTime":41,"deleteTime":30},"前端开发","frontend-engineering","2222","2026-04-03T02:36:11.945Z","2026-04-07T16:38:46.496Z",[43],{"articleId":23,"tagId":44,"createTime":33,"tag":45},"2",{"id":44,"tagName":46,"slug":46,"themeColor":47,"description":48,"createTime":49,"updateTime":50,"deleteTime":35},"Vue3","#00d9ff","记录学习Vue3","2026-04-03T02:42:17.413Z","2026-04-03T02:42:38.863Z",[],[]]