Vue3笔记

news/2024/10/8 6:29:58

本节代码可以从git clone https://gitee.com/fullstacker1024/vue3-component-demo.git下载
其中项目实例todo

1、前言

1)Vue3的优势

(1)更容易维护
①组合式API
②更好的TypeScript支持
(2)更小的体积
①良好的TreeShaking
②按需引入
(3)更快的速度
①重写diff算法
②模板编译优化
③更高效的组件初始化
(4)更优的数据响应式
proxy

2)create-vue搭建Vue3项目

create-vue是Vue官方新的脚手架工具,底层切换到了vite(下一代构建工具),为开发提供极速响应。

(1)前提环境条件
已安装16.0或更高版本的Node.js
(2)创建一个Vue应用
npm init vue@latest这一指令将会安装并执行create-vue
运行项目:

npm install
npm run dev

3)项目目录和关键文件

image

vue2插件是Vetur,vue3插件是volar

2、组合式API

1)setup选项

(1)setup选项的的写法和执行时机
image
(2)setp选项中写代码的特点
image
(3)<script setup/>语法糖
image

2)reactive和ref函数

(1)reactive()
作用:接受对象类型数据的参数传入并返回一个响应式的对象
核心步骤:

从vue包中导入reactive函数
<script setup/>中执行reactive函数并传入类型为对象的初始值,并使用变量接收返回值

<script setup>
import {reactive} from 'vue'
const state =reactive(对象类型数据)
</script>

(2)ref()
作用:接收简单类型或对象类型的数据传入并返回一个响应式对象
核心步骤:

从vue包中导入ref函数
<script setup/>中执行ref函数并传入初始值,使用变量接收ref函数的返回值

<script setup>
import {ref} from 'vue'
const count =ref(对象类型数据)
</script>

注意点:

脚本中访问数据,需要通过.value
在template中,.value不需要加

3)computed

计算属性的基本思想和Vue2的完全一致,组合式API下的计算属性只是修改了写法。
核心步骤:

导入computed函数
执行函数在回调参数中return基于响应式数据做计算的值,用变量接收

image
示例:

点击查看代码
<script setup>
import { ref, computed } from 'vue'
const list = ref([1, 2, 3, 4, 5, 6, 7, 8, 9])
const computedList = computed(() => {return list.value.filter(item=>item>5)
})const addFn=()=>{var randomNumber = Math.floor(Math.random() * 10) + 1;list.value.push(randomNumber)
}
</script><template><div>{{ list }}</div><div>>5:{{ computedList }}</div><div><button @click="addFn">增加随机数</button></div>
</template><style scoped></style>

计算属性中不应该有“副作用”:比如异步请求/修改dom
避免直接修改计算属性的值:计算属性应该是只读的,特殊情况可以配置get/set

4)watch

作用:侦听一个或者多个数据的变化,数据变化时执行回调函数。
两个额外参数:1.immediate(立即执行)2.deep(深度侦听)

(1)侦听单个数据
①导入watch函数
②执行watch函数传入要侦听的响应式数据(ref对象)和回调函数

点击查看代码
<script setup>
import { watch, ref } from 'vue'
const count = ref(0)watch(count, (newValue, oldValue) => {console.log(`count发生了变化,旧值为${oldValue},新值为${newValue}`);
})
</script>

(2)侦听多个数据
同时侦听多个响应式数据的变化,不管哪个数据变化都需要侦听

点击查看代码
<script setup>
import { watch, ref } from 'vue'
const count = ref(0)
const name = ref('cp')watch([count, name], ([newCount, newName], [oldCount, oldName]) => {console.log('count或者name发生了变化', [newCount, newName], [oldCount, oldName]);
})
const changeCount=()=>{count.value++
}</script><template><div>{{count}}-{{ name }}</div><div><button @click="changeCount">watch</button></div>
</template><style scoped></style>

(3)immediate
立刻执行

点击查看代码
<script setup>
import { watch, ref } from 'vue'
const count = ref(0)watch(count, (newValue, oldValue) => {console.log(`count发生了变化,旧值为${oldValue},新值为${newValue}`);
},{
immediate:true
})
</script>

(4)deep
深度监视,默认watch进行的是浅层监视

const ref1=ref(简单类型) //可以直接监视
const ref2=ref(复杂类型)//监视不到复杂类型内部数据的变化

点击查看代码
<script setup>
import { watch, ref } from 'vue'
const userInfo = ref({name: 'zs',age: 18
})watch(userInfo, (newValue) => {console.log(newValue)
},{deep:true
})
const changeUserInfo = () => {userInfo.value.name = 'lisi'
}</script><template><div>{{ userInfo }}</div><div><button @click="changeUserInfo">watch</button></div>
</template><style scoped></style>

(5)精确侦听对象的某个属性
在不开启deep的前提下,侦听age的变化,只有age变化时才知晓回调。
image

5)生命周期函数

image

6)父子通信-defineProps、defineEmits

(1)父传子
基本思想:

父组件中给子组件绑定属性
子组件内部通过props选项接收defineProps编译器宏
defineProps原理:就是编译阶段的一个标识,实际编译器解析时,遇到后悔进行编译转换

image

父组件传过来的数据在script中需要使用props.message形式调用
父组件传过来的数据在template中直接调用:message
动态值使用:绑定

(2)子传父

父组件中给子组件标签通过@绑定事件
子组件内部通过emit方法触发事件defineEmits

image

7)模板引用-defineExpose

通过ref标识获取真实的dom对象或者组件实例对象

调用ref函数生成一个ref对象
通过ref标识绑定ref对象到标签

image
(1)defineExpose()
默认情况下<scritp setup>语法糖下组件内部的属性和方法是不开放给父组件访问的,可以通过defineExpose编译宏指定哪些属性和方法容许访问
image

8)provide和inject

作用和场景:顶层组件向任意的底层组件传递数据和方法,实现跨层组件通信。

顶层组件通过provide函数提供数据
底层组件通过inject函数获取数据

(1)跨层传递普通数据
provide第二个参数是普通数据

//top.vue
import { provide } from 'vue';
//普通数据
provide('theme-color', 'pink')//bottom.vue
import { inject } from 'vue';
const themeColor =inject('theme-color')

(2)跨层传递响应式数据
provide第二个参数是ref对象

//top.vue
import { provide, ref } from 'vue';
//响应式数据
const count = ref(100)
provide('count', count)//bottom.vue
import { inject } from 'vue';
const count=inject('count')

(3)跨层传递方法
顶层组件可以向底层组件传递方法,底层组件调用方法修改顶层组件中的数据。
provide第二个参数是函数

//top.vue
//传递方法
import { provide, ref } from 'vue';
const count = ref(100)
provide('changeCount', (newCount) => {count.value = newCount
})//bottom.vue
import { inject } from 'vue';
const changeCount=inject('changeCount')
const ChangeFn=()=>{changeCount(5)
}

3、Vue3.3新特性

1)defineOptions

背景说明:

有< script setup/>之前,如果要定义props,emits可以轻而易举地天剑一个与setup平级的属性。但是用了< script setup/>后,就没法这么干了,setup属性已经没有了,自然无法添加与其平级的属性。

为了解决这一问题,引入了definePropsdefineEmits这两个宏。但这只解决了props与emits这两个属性。如果我们要定义组件的name或其他自定义的属性,还是得回到最原始的用法:再添加一个普通的< script>标签。这样就会存在两个< script>标签。让人无法接受。

所以在Vue3.3中新引入了defineOptions宏。顾名思义,主要是来定义Options API的选项。可以用defineOptionis定义任意的选项,props、emits、expose、slots除外(因为这些可以使用defineXXX来做到)。

<script setup>
defineOptions({name: 'Foo',inheritAttrs:false,//...更多自定义属性
})
</script>

2)defineModel

在Vue3中,自定义组件上使用v-model,相当于传递一个modelValue属性,同时触发update:modelValue事件。

<Child v-model="isVisible">
//相当于
<Child :modelValue="isVisible" @update:modelValue="isVisible=$event">

①不使用defineModel
我们需要先定义props,再定义emits。其中有许多重复的代码。如果需要修改此值,还需要手动调用emit函数。
示例代码:

点击查看代码
//父组件
<script setup>
import noDefineModelComponent from '@/components/defineModel/noDefineModelComponent.vue'
import {ref} from 'vue'const txt=ref('1')</script><template>
<div>{{ txt }}<noDefineModelComponent v-model="txt"></noDefineModelComponent>
</div>
</template><style scoped></style>//子组件
<script setup>
defineProps({modelValue: String
})const emit = defineEmits(['update:modelValue'])
</script><template><div><input type="text" :value="modelValue" @input="e => emit('update:modelValue', e.target.value)"></div>
</template><style scoped></style>

②使用defineModel
需要先在vite.config.js中开启defineModel配置

plugins: [vue({script: {defineModel: true,},}),],

示例代码:

点击查看代码
//父组件
<script setup>
import defineModelComponent from '@/components/defineModel/defineModelComponent.vue'import {ref} from 'vue'const txt=ref('1')</script><template>
<div>{{ txt }}<defineModelComponent v-model="txt"></defineModelComponent></div>
</template><style scoped></style>//子组件
<script setup>
import { defineModel } from 'vue'
const modelValue = defineModel()
</script><template><div><input type="text" :value="modelValue" @input="e => modelValue= e.target.value"></div>
</template >
<style scoped></style>

4、Pinia

Pina是Vue的最新的状态管理工具是Vuex的替代品。

提供更加坚定的API(去掉了mutation)
提供符合组合式风格的API(和Vu3新语法统一)
去掉了modules的概念,每一个store都是一个独立的模块
配合 TypeScript更加友好,提供可靠的类型推断

1)手动添加Pinia到Vue项目

在实际开发项目的时候,关于Pinia的配置,可以在项目创建时自动添加。
现在我们手动添加下:
①使用vite创建一个空的vue3项目

npm create vue@latest

②按照官方文档安装pinia到项目中

import { createApp } from 'vue'
import { createPinia } from 'pinia'
import App from './App.vue'const pinia =createPinia()
createApp(App).use(pinia).mount('#app')

定义了一个store,该store在调用之前,store实例是不会被创建的。
store是一个用reactive包装的对象,这意味着不需要再getters后面写.value,就像setup中的props一样,如果写了,也不能解构它。

2)Pinia基本语法-state、action、getters

(1)定义
新建store/couter.js文件:
示例:

import { defineStore } from "pinia";
import { computed, ref } from "vue";
//定义store
export const counterStore = defineStore("counter", () => {//声明数据 stateconst count = ref(0);//声明操作数据的方法 actionconst addCount = () => count.value++;const subCount = () => count.value--;//声明基于数据派生的计算数学 gettersconst double=computed(()=>count.value*2)const msg = ref("hello pinia");return {count,addCount,subCount,double,msg,};
});

(2)使用:
导入couter.js后,可以在scritp或者template中直接使用。
示例:

<script setup>
import { counterStore } from '@/store/counter';const userCounterStore=counterStore()
</script><template><div class="parent"><h3>ParentApp-{{userCounterStore.count}}-{{ userCounterStore.msg }}-{{ userCounterStore.double }}</h3></div>
</template><style scoped>
.parent {border: 1px solid #000;padding: 30px;
}</style>

3)action异步写法

异步action函数的写法和组件中获取异步数据的写法完全一致。

4)storeToRefs写法

为了从store中提取属性(作为action的方法除外,可以直接解构)时保持其响应性,你需要使用storeToRefs()。它将为每一个响应式属性创建引用。

直接解构,不处理,数据会丢失响应式

<script setup>
//counter.js见上节
import { useChannelStore } from '@/store/counter';const channelStore=useCounterStore()
const {count ,msg}=counterStore
</script>

使用storeToRefs

<script setup>
//counter.js见上节
import { useChannelStore } from '@/store/counter';
import {storeToRefs} from 'pinia'const channelStore=useCounterStore()
const {count ,msg}=storeToRefs(counterStore)
</script>

5)持久化

Pinia持久化插件:
(1)使用步骤
①安装插件pinia-plugin-persistedstate
npm i pinia-plugin-persistedstate
②main.js使用
将插件添加到pinia实例上

import persist from 'pinia-plugin-persistedstate'
//...
app.use(CreatePinia().use(persist))

③store仓库中,persist:true(选项式)开启

以上步骤,整改store将使用默认持久化配置保持。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hjln.cn/news/42611.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈,一经查实,立即删除!

相关文章

视野修炼-技术周刊第87期 | Nodejs 15 周年

① js 中的经典八股 ② merge-anything - 深度合并对象的库 ③ DOM 树深度对渲染性能的影响 ④ Node.js 15周年 ⑤ 页面元素测量工具 ⑥ Best Cookier - cookie管理插件 ⑦ ShotEasy - 在线的图片压缩 ⑧ 2024 应该了解的 Node.js 特性 ⑨ CSS 长度单位介绍 - 非常全面的指南 ⑩…

Diffusion模型

参数说明 beta_schedule = np.linspace(0.0001, 0.02, 1000) # 示例 beta schedulealpha_hat = np.cumprod(1 - beta_schedule) # 计算 alpha_hat具体例子 让我们通过一个具体的例子展示如何计算 Beta Schedule 和 Alpha_hat: import numpy as np# 定义 Beta Schedule beta_…

NSCC集群使用笔记

1. 账号申请 如果是 NUS,NTU 或者 ASTAR 的学生,可以直接用自己的学校 ID 登录。登录不上的话可以发邮件联系 nscc 工作人员即可,基本上第二天就会回复解决。 2. VSCode 连接 账号申请下来后进官网设置你的 ssh key 之类的东西就可以登录了。第一次登录成功后,可以参考这篇…

OOP4-6作业总结性Blog

这三次大作业经历了一次题型的更新,不管过程是怎么说的,结果是试卷批改程序退出出题的视野了,取而代之的是电路设计题。对于电路设计题我认为出的更加合理些,不像试卷批改题难点都在写正则表达式上,多了对类间关系的考察。 这三次题目集期间,我们在课上主要学习了一系列设…

.NET借助虚拟网卡实现一个简单异地组网工具

由于工作需要,经常需要远程客户的服务器,但是并不是所有服务器都能开外网端口,使用向日葵等软件终究还是不太方便,于是找了很多工具,包括zerotier 等,但是由于服务器在国外等有时候还不同, 于是开始自己想办法研究一个属于自己的组网工具,最后找到snltty大佬的 https:/…

BUUCTF-Misc(131-140)

[ACTF新生赛2020]剑龙 打开pwd.txt发现是颜文字然后打开随波逐流,AAencode颜文字解密得到welcom3! 看一下这个图片的详细信息,发现然后用颜文字结出来的那个密码,去steghide解密U2FsdGVkX1/7KeHVl5984OsGUVSanPfPednHpK9lKvp0kdrxO4Tj/Q==又是U2f然后这次我还以为是AES加密,…

2024-06-09 闲话

2024-06-09 闲话看老友记看到这里。 诶这个名字好熟悉,诶这个小人物也好熟悉。 诶 Bidirectional Encoder Representations from transformers 也是 BERT 诶我草,这是芝麻街。

[AI资讯0609] SamAltman建立了庞大投资帝国,通义千问Qwen2发布即爆火,OpenAI泄密者公布165页文件,奥特曼百万年薪挖角谷歌TPU人才……

SamAltman建立了庞大投资帝国,但不持OpenAI股票;个人资产包括房产和多家公司股权。他曾在YCombinator担任总裁,并投资40家公司,有五家估值增长100倍以上。尽管他不参与OpenAI决策,但其利益与公司业务存在复杂关系,引发监督和透明度争议……AI资讯奥特曼28亿「投资帝国」曝…