首页 > 其他分享 >Vue3笔记

Vue3笔记

时间:2024-06-09 22:12:51浏览次数:27  
标签:count vue const 笔记 Vue3 组件 import ref

本节代码可以从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", () => {
  //声明数据 state
  const count = ref(0);
  //声明操作数据的方法 action
  const addCount = () => count.value++;
  const subCount = () => count.value--;
  //声明基于数据派生的计算数学 getters
    const 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将使用默认持久化配置保持。

标签:count,vue,const,笔记,Vue3,组件,import,ref
From: https://www.cnblogs.com/hoaprox/p/18236004

相关文章

  • Vitis HLS 学习笔记--static RAM/ROM
    目录1.简介2.staticRAM2.1无reset的情形2.2含reset的情形3.staticROM4.总结1.简介本文仍然是讨论阵列的初始化与复位问题,区别于《VitisHLS学习笔记--global_array_RAM初始化及复位-CSDN博客》,本文讨论的对象是静态阵列,RAM和ROM。静态阵列可以映射到B......
  • Linux笔记
    (1)常用命令: [1]Ctrl: +a 移动至行首 +e 移动至行尾 +w 清除光标之前一个单词 +k 清除光标到行尾的字符 +h 删除光标前一个字符 +b 光标前移一个字符 +f 光标后移一个字符 +c 取消本次命令运行 +d 退出/登出(vi/vim除外) +l 清屏 (=命令clear) +r 匹配历史......
  • python学习笔记
    (1)help('keywords')控制台内输入关键字 pycharm常用操作: [1]ctrl+y 恢复上一次操作 [2]ctrl+z 撤回上一次操作(2)%常见格式: [1]%s 字符串 [2]%d 整数 [3]%f 小数(浮点数)(3)注释: 单行注释: #________ 多行注释: """______"""(4)运算符号: / 除 // ......
  • NSCC集群使用笔记
    1.账号申请如果是NUS,NTU或者ASTAR的学生,可以直接用自己的学校ID登录。登录不上的话可以发邮件联系nscc工作人员即可,基本上第二天就会回复解决。2.VSCode连接账号申请下来后进官网设置你的sshkey之类的东西就可以登录了。第一次登录成功后,可以参考这篇文章设置ssh......
  • Xinhui学习NLP的笔记本:基于MLP/CNN的姓氏分类系统
    ASurnameClassificationSystembasedonMLP基于MLP的姓氏分类系统ThisnotebookservesasmylearningjourneyintotheMultilayerPerceptron(MLP),whichisafundamentaltypeofFeedforwardNeuralNetwork.Throughoutthisarticle,Iwillbeundertakin......
  • 「笔记」模电复习 2
    信号的处理与变换频率特性放大电路中由于电抗元件等的存在,输入信号频率过低或过高时,放大倍数变小,且产生相移,即放大倍数是信号频率的函数。此函数关系称为频率响应或频率特性。高通电路。输入频率越高,输出电压越接近输入电压。\(\dotA_u=\frac{\dotU_o}{\dotU_i}=\fra......
  • 反悔贪心学习笔记
    算法:反悔贪心,顾名思义就是贪心的时候反悔。意思是:如果这一步的贪心不是全局最优解,就退回去一步,换一种贪心策略。一般分为反悔自动机和反悔堆。反悔自动机基本的思路是:每次选择直观上最接近全局最优解的贪心策略,若发现最优解不对,就想办法自动支持反悔策略。反悔堆则......
  • Java Web学习笔记29——Vue路由
    Vue路由:前端路由:点击菜单栏,地址栏会发生变化,会显示对应的组件。URL中的Hash(#号后面的部分)与组件之间的对应关系。Hash是/dept,那么就是部门管理组件;Hash是/emp,那么就是员工管理组件;VueRouter:介绍:VueRouter是Vue的官方路由;组成:1)VueRouter:路由器类,根据路由请求在路......
  • 计算机网络个人笔记
    ARP过程简易叙述想要与对端主机通信首先查看本地高速缓存表中是否有到对端主机的地址,如果本地缓存未老化或者自清除,有则直接让网关转发;没有地址则开始进行ARP广播向网关请求已知通信IP地址的Mac地址。网关收到ARP请求地址查询本地路由表为其进行下一步,如果本地路由表没有查到请......
  • RabbitMQ笔记
     端午无聊,就来学学MQ吧消息队列消息指的是两个应用之间传递的数据消息队列是在消息传递中保存消息的容器生成者:只负责发送数据消费者:只负责读取数据(数据就是消息)为什么要用消息队列解耦如果一个系统(系统a)的需求是共享自己系统的数据,而其他系统(系统BCD)是需求者。而......