Vue3(二)计算属性Computed,监视属性watch,watchEffect,标签的ref属性,propos属性,生命周期,自定义hook

文章目录

  • 一 、计算属性
    • 1. 简写
    • 2. 完整写法
  • 二、监视watch
    • 1. 监视【ref】定义的【基本类型】数据
    • 2. 监视【ref】定义的【对象类型】数据
    • 3. 监视【reactive】定义的【对象类型】数据
    • 4. 监视【ref】或【reactive】定义的【对象类型】数据中的某个属性
    • 5. 监视多个数据
    • 总结
  • 三、watchEffect
  • 四、标签的ref属性
  • 五、Propos
    • 1.子组件接收,接收并保存
    • 2. 限制接收的类型,限制必要性
    • 3. 子组件给默认值
  • 六、生命周期
  • 七、自定义Hook

一 、计算属性

computed:根据已有数据计算出新数据(和Vue2中的computed作用一致).
在这里插入图片描述
还是那个案例,当输入框里的内容变化时,全名跟着修改。

1. 简写

语法:computed(()=>{....})

// 引入computed 
import { ref, computed } from "vue"let firstName = ref("zhang")
let lastName = ref("san")// 这么定义的fullName是一个计算属性,且是只读的(也就是简写形式)
let fullName = computed(() => {return firstName.value + "-" + lastName.value
})

2. 完整写法

语法:computed({ get() {...}, set(val) {... }, })

给按钮绑定事件,修改姓名

 <button @click="changeFullName">全名改为:li-si</button>

在这里插入图片描述
代码:

let fullName = computed({get() {return firstName.value + "-" + lastName.value},set(val) {const [str1, str2] = val.split("-")firstName.value = str1lastName.value = str2},
})

二、监视watch

  • 作用:监视数据的变化(和Vue2中的watch作用一致)
  • 特点:Vue3中的watch只能监视以下四种数据
  1. ref定义的数据(基本数据类型,对象类型)。
  2. reactive定义的数据。
  3. 对象类型数据里的某个属性。
  4. 一个包含上述内容的数组。

1. 监视【ref】定义的【基本类型】数据

语法watch(监视对象,回调函数)
监视对象直接写变量名即可,监视的是其value值的改变。

import { ref, watch } from "vue"
// 数据
let sum = ref(0)
// 方法
function changeSum() {sum.value += 1
}
// 监视 watch(监视对象,回调函数);
const stopWatch = watch(sum, (newValue, oldValue) => {if (newValue > 10) {stopWatch()}
})

如果想停止监视,则接收watch的返回值,watch返回的值是一个函数stopWatch,调用该函数就停止监视sum

2. 监视【ref】定义的【对象类型】数据

在这里插入图片描述

import { ref, watch } from "vue"
// 数据
let person = ref({name: "tom",age: 18,
})
function changeName() {person.value.name += "~"
}
function changeAge() {person.value.age += 1
}
function changePerson() {person.value = { name: "李四", age: 50 }
}
watch(person, (newValue, oldValue) => {console.log("person变化", newValue, oldValue)
})

点击这三个按钮发现,只有当点击修改整个人时,watch监视才起作用。因为对于ref定义的对象数据类型,watch监视的是地址值。若想监视对象里面的值,则开启深度监视

watch(person,(newValue, oldValue) => {console.log("person变化", newValue, oldValue)},{ deep: true } // 也可配置 immediate: true
)

另外,点击前两个按钮发现newValueoldValue是同一个值。点击修改整个人时,两个值才不一样。
在这里插入图片描述
因为修改名字和年龄只是修改了对象的属性,而修改整个人是修改了对象的地址。地址不一样了,newValueoldValue值也就不一样了。

总结:

  1. 语法:watch(监视对象,回调函数,{配置对象deep、immediate等等.....})
  2. 监视【ref】定义的【对象类型】数据时,监视的是对象的地址值,若想监视对象内部属性的变化,需要手动开启深度监视
  3. 若修改的是ref定义的对象中的属性,newValueoldValue 都是新值,因为它们是同一个对象。
    若修改整个ref定义的对象,newValue 是新值, oldValue 是旧值,因为不是同一个对象了。

3. 监视【reactive】定义的【对象类型】数据

把上边的代码改了改:
在这里插入图片描述
点击三个按钮,控制台打印的结果:
在这里插入图片描述

  1. 不用写deep:true,默认开启了深度监听。并且这个深度监听是不可关闭的。
  2. 注意到这三行打印的结果里newValueoldValue都是一样的值,这是因为person对象的地址始终没有发生变化。

总结:

监视reactive定义的【对象类型】数据时,默认开启了深度监视且无法关闭。

4. 监视【ref】或【reactive】定义的【对象类型】数据中的某个属性

上边都是监视一个对象,这里监视对象的某属性。
在这里插入图片描述

let person = reactive({name: "张三",age: 18,car: {c1: "奔驰",c2: "宝马",},
})
// 方法
function changeName() {person.name += "~"
}
function changeAge() {person.age += 1
}
function changeC1() {person.car.c1 = "奥迪"
}
function changeC2() {person.car.c2 = "大众"
}
function changeCar() {
// reactive对象里属性对象可以这样改person.car = { c1: "雅迪", c2: "爱玛" }
}

只监视对象里的某一个基本数据类型age

// 第一个参数写成函数的形式,指定监听的属性;
// 如果写的是person,则无论点击哪个按钮,都会被监听到
watch(() => person.age,  (newValue, oldValue) => {console.log("age改变", newValue, oldValue) // age改变 19 18}
)

监视对象里的对象类型属性car

watch(() => person.car,(newValue, oldValue) => {console.log("person.car变化了", newValue, oldValue)},{ deep: true }
)

不写deep:true,则监视的是car这个对象,当其地址发生变化时才会执行console.log打印(也就是不会默认开始深度监听)。
deep:true,则也监视到car内部的属性c1,c2。属性或对象发生变化都会打印
在这里插入图片描述

总结:

  1. 监视响应式对象中的某个属性,且该属性是基本类型的,要写成函数式
  2. 监视响应式对象中的某个属性,且该属性是对象类型的,可以直接写,也能写函数,更推荐写函数
  3. 监视reactive响应式对象中的某个属性,且该属性是对象类型的(car),不会默认开始深度监听

5. 监视多个数据

还是4中的案例,需求是监视人的姓名和车

watch([() => person.name, person.car],(newValue, oldValue) => {console.log("person.car变化了", newValue, oldValue)},{ deep: true }
)

总结

  1. 对于ref定义的基本数据类型,监视的是其value
  2. 对于ref定义的对象数据类型,监视的是对象的地址值,若要监视对象里的属性,需要手动开启深度监视
  3. 对于reactive定义的对象数据类型,默认开启了深度监视
  4. 对于refreactive对象类型里的某个属性,如果是该属性是对象(car),监视的是其地址值,不会默认开始深度监视

三、watchEffect

watch对比watchEffect
(1) 都能监听响应式数据的变化,不同的是监听数据变化的方式不同
(2) watch:要明确指出监视的数据
(3) watchEffect:不用明确指出监视的数据(函数中用到哪些属性,就监视哪些属性)

在这里插入图片描述

import { ref, watch, watchEffect } from "vue"
// 数据
let temp = ref(0)
let height = ref(0)// 方法
function changeTemp() {temp.value += 10
}
function changeHeight() {height.value += 10
}// 用watch实现
watch([temp, height], (val) => {// 从value中获取最新的temp值、height值let [newTemp, newHeight] = valif (newTemp >= 50 || newHeight >= 20) {console.log("发送请求")}
})// 用watchEffect
watchEffect(() => {if (temp.value >= 50 || height.value >= 20) {console.log("发请求")}
})

四、标签的ref属性

ref用于给标签打标识。
1. 用在普通DOM标签上

<template><div class="person"><h2 ref="title">hello</h2><h2 ref="title2">hello again</h2><h2>world</h2><button @click="showTitle">获取所有hello</button></div>
</template><script lang="ts" setup name="Person">
import { ref } from "vue"
// 似乎是解构赋值,这里接收的变量名需要和ref定义的一致
let title = ref()
let title2 = ref()
// let title1 = ref()  这样打印出title1是undefined
function showTitle() {console.log('ref', ref());console.log('title', title.value)console.log('title2', title2.value)
}
</script>

变量名需要的ref的名称一致,否则undefined
在这里插入图片描述
(这个ref打印出来也不知道是什么东西,看不懂,有点子神奇了)

2. 用在组件标签上
(竟然可以实现子组件给父组件传值)

父组件

<template><Person ref="ren" /><button @click="showInfo">点击获取组件信息</button>
</template><script lang="ts" setup name="APP">
import Person from "./components/Person.vue"
import { ref } from "vue"
let ren = ref()
function showInfo() {console.log(ren.value.name)console.log(ren.value.age)
}
</script>

子组件

// 不引入也行,definexxx都是一些宏函数,会自动引入
import { defineExpose } from "vue"
let name = "tom"
let age = 18
// 使用defineExpose将组件中的数据交给外部
defineExpose({ name, age })

五、Propos

propos用于父组件向子组件传值

父组件的数据:

import Person from './components/Person.vue'
import { reactive } from 'vue';
import { Persons } from './type';
// 父组件
let persons = reactive<Persons>([{ id: '12345', name: 'tom', age: 18 },{ id: '123456', name: 'jerry', age: 18 },{ id: '1234567', name: 'lucy', age: 18 }
])

1.子组件接收,接收并保存

<!--父组件--><Person :list=persons />
<!--子组件-->
<script>// 只接收defineProps(['list'])console.log(list[0]); // 没保存,直接读取会提示“找不到list”// 接收并保存let x = defineProps(['list'])console.log(x.list[0]); // Proxy(Object) {id: '12345', name: 'tom', age: 18}
</script>

2. 限制接收的类型,限制必要性

<!--父组件-->
<Person :list=persons /><Person :list="5" />  <!-- 不是Person类型会报错 --><Person />            <!-- 没有给子组件传递数据也会报错,解决方法:添加`?` --><!--子组件-->
<script>// 接收+限制类型,只能接收Persons类型的数据defineProps<{ list: Persons }>()
</script>

当添加一个?之后,则父组件可传可不传。不传递也不会报错

 defineProps<{ list?: Persons }>()

3. 子组件给默认值

import { withDefaults } from 'vue';
// 接收+限制类型+限制必要性+指定默认值
withDefaults(defineProps<{ list?: Persons }>(), {list: () => [{ id: '默认', name: '默认', age: 10 }]
})

六、生命周期

回顾vue2的生命周期,vue2(三) vue生命周期

Vue2的生命周期

创建阶段:beforeCreatecreated

挂载阶段:beforeMountmounted

更新阶段:beforeUpdateupdated

销毁阶段:beforeDestroydestroyed

Vue3的生命周期

创建阶段:setup

挂载阶段:onBeforeMountonMounted

更新阶段:onBeforeUpdateonUpdated

卸载阶段:onBeforeUnmountonUnmounted

常用:onMounted(挂载完毕),onUpdated(更新完毕),onBeforeUnmount(卸载之前)

解释一下这个创建阶段,用代码表示就是:

<script lang="ts" setup name="Person">
import { onBeforeMount,  onMounted,  } from 'vue';console.log('创建');onBeforeMount(() => {console.log('挂载之前');
})onMounted(() => {console.log('挂载完毕');
})
</script>

在这里插入图片描述

七、自定义Hook

一直在说vue3是组合式API,每个功能的数据,方法都放在一起。目前在<script>标签里还是上边写一堆功能的数据,下面写一堆功能的方法,和vue2没啥区别了。
hook其实就是将各个功能的数据和方法单独封装到一个hook文件中。以后修改功能就直接去这个hook文件里改就好了。

  • 什么是hook?—— 本质是一个函数,把setup函数中使用的Composition API进行了封装,类似于vue2.x中的mixin
  • 自定义hook的优势:复用代码, 让setup中的逻辑更清楚易懂。

现在页面上有两个功能:
在这里插入图片描述
src/hook/useSum.ts

import {  ref, computed } from 'vue'
export default function(){
// 功能一的数据let sum = ref(0)
// 功能一的函数let bigSum = computed(() => {return 10 * sum.value})function addSum() {sum.value += 1}return {sum,addSum,bigSum}
}

src/hook/useDog.ts

import { reactive, ref, computed } from 'vue'
import axios from 'axios';
export default function(){
// 功能二的数据
let dogList = reactive<string[]>([])
// 功能二的函数
async function getDog() {try {// 发请求let { data } = await axios.get('https://dog.ceo/api/breed/pembroke/images/random')// 维护数据dogList.push(data.message)} catch (error) {// 处理错误console.log(error.message)}
}
return {dogList,getDog}
}

组件内:
src/components/Person.vue

<template><div class="person"><h2> 求和为{{ sum }},扩大十倍后{{ bigSum }}</h2><button @click="addSum">sum+1</button><br><br><img v-for="(u, index) in dogList" :key="index" :src="(u as string)"><br><button @click="getDog">再来一只狗</button></div>
</template><script lang="ts" setup name="Person">
// 1. 引入
import useDog from '../hook/useDog.ts'
import useSum from '../hook/useSum.ts'
// 2. 解构获取信息
const { dogList, getDog } = useDog()
const { sum, addSum, bigSum } = useSum()
</script>

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

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

相关文章

Android下MVP和MVVM模式的实践

转载注明出处&#xff1a;https://blog.csdn.net/skysukai 1、前言 MVP和MVVM诞生已经好些年头了&#xff0c;记得刚毕业才参加工作的时候&#xff0c;第一次见到了有上万行的Activity&#xff0c;这种巨无霸的Activity维护起来简直就是噩梦。这时候&#xff0c;就需要进行代…

2024最新windows 11系统 PHP或者idea编译器-配置Git环境和使用教程

文章目录 目录 文章目录 安装流程 小结 概要安装流程技术细节小结 概要 确保电脑上已安装到git,如下图所示&#xff1a;-是已安装好&#xff1a; 安装git教程&#xff1a; Git安装使用教程_git安装教程-CSDN博客 安装流程 点击左上角如图所示&#xff1a; 需要验证git本地 …

matlab恢复默认窗口布局

1.点击主页&#xff0c;选择布局 2.选择默认&#xff0c;即可恢复到默认的窗口布局

ollama 部署教程(window、linux)

目录 一、官网 二、安装方式一&#xff1a;window10版本下载 三、安装方式二&#xff1a;linux版本docker 四、 模型库 五、运行模型 六、API服务 七、python调用 ollama库调用 langchain调用 requests调用 aiohttp调用 八、模型添加方式 1.线上pull 2.导入 GGU…

类中的特殊内容

仿照string类&#xff0c;自己手动实现 My_string #include <iostream> #include <string.h> using namespace std;class My_string { private:int len;int size;char *ptr; public:My_string():size(15),len(0){ptrnew char[size];ptr[0]\0;}My_string(const char…

拓维思注册机Tovos PowerLine4.0.19树障分析 Tovos SmartPlan2.0.0航线规划软件

Tovos PowerLine是功能强大的输电线路智能巡检系统&#xff01;这是一个专业且智能的软件&#xff0c;能够更准确的进行巡检和对线路设备进行精确的测量&#xff0c;通过获取高精度的点云来获取精准的三维路线的地形地貌、设备设施、途径的各种物体等来精确您的三维空间信息和三…

【初阶数据结构】详解二叉树 - 树和二叉树(三)(递归的魅力时刻)

文章目录 前言1. 二叉树链式结构的意义2. 手搓一棵二叉树3. 二叉树的遍历&#xff08;重要&#xff09;3.1 遍历的规则3.2 先序遍历3.3 中序遍历3.4 后序遍历3.5 遍历的代码实现3.5.1 先序遍历代码实现3.5.2 中序遍历代码实现3.5.3 后序遍历代码实现 4. 统计二叉树结点的个数5.…

基于Vision-Board的智能应急环境监测控制小车

目录 1 项目概述 1.1 项目背景 1.2 系统功能介绍 1.2.1 下位机智能小车控制系统 1.2.2 微信小程序App 1.2.3 PC上位机App 1.3 框图介绍 1.3.1 主控板卡 1.3.2 小车控制模块 1.3.3 通信模块 1.4 系统使用的技术要点 2 系统硬件设计 2.1 Version board主控板块系统结…

《深度学习》卷积神经网络CNN 实现手写数字识别

目录 一、卷积神经网络CNN 1、什么是CNN 2、核心 3、构造 二、案例实现 1、下载训练集、测试集 代码实现如下&#xff1a; 2、展示部分图片 运行结果&#xff1a; 3、图片打包 运行结果&#xff1a; 4、判断当前使用的CPU还是GPU 5、定义卷积神经网络 运行结果&a…

通信工程学习:什么是NFVO网络功能虚拟化编排器

NFVO&#xff1a;网络功能虚拟化编排器 NFVO&#xff08;Network Functions Virtualization Orchestrator&#xff09;&#xff0c;即网络功能虚拟化编排器&#xff0c;是网络功能虚拟化&#xff08;NFV&#xff09;架构中的核心组件之一。NFV是一种将传统电信网络中的网络节点…

Linux学习笔记13---GPIO 中断实验

中断系统是一个处理器重要的组成部分&#xff0c;中断系统极大的提高了 CPU 的执行效率&#xff0c;本章会将 I.MX6U 的一个 IO 作为输入中断&#xff0c;借此来讲解如何对 I.MX6U 的中断系统进行编程。 GIC 控制器简介 1、GIC 控制器总览 I.MX6U(Cortex-A)的中断控制器…

全栈开发(三):springBoot3中使用mybatis-plus

MyBatis-Plus &#x1f680; 为简化开发而生 (baomidou.com) 1.配置pom.xml <dependency><groupId>com.baomidou</groupId><artifactId>mybatis-plus-spring-boot3-starter</artifactId><version>3.5.7</version></dependency&g…

90%的爆文作者都在用的AI标题公式 超实用7个迭代技巧

本文背景 我的上篇文章&#xff0c;关于我是如何在5分钟之内写出AI爆文结构化提示词的&#xff08;附50条优化指令词&#xff09;&#xff0c;已经详细的讲解了如何快速生成提示词&#xff0c;以及一些常用的优化提示词的指令&#xff0c;今天大象再来详细掰头掰头如何迭代提示…

虚拟摄像头抓屏

目录 一、下载: 二、安装 三、使用 前两天跟客户闲聊,说的了一个应用需求。他想实现将服务器操作过程实时记录下来,好比现在很多博主拍摄Vlog,再具体一点儿就是维修类短视频,可以记录维修过程,发现错误可以参照视频恢复,成功了也可以作为日后培训的教程。 实现的方法…

第一个Web项目(java+servlet+jsp)

通过百度网盘分享的文件&#xff1a;第一个Web项目 链接&#xff1a;https://pan.baidu.com/s/11vnAPeAf6Dtax7H6aYKZgA 提取码&#xff1a;1234 目录 声明&#xff1a; 简介&#xff1a; 注意&#xff1a; 操作步骤&#xff1a; 1.在idea中新建java项目&#xff0c;项目…

手写数字识别案例分析(torch,深度学习入门)

在人工智能和机器学习的广阔领域中&#xff0c;手写数字识别是一个经典的入门级问题&#xff0c;它不仅能够帮助我们理解深度学习的基本原理&#xff0c;还能作为实践编程和模型训练的良好起点。本文将带您踏上手写数字识别的深度学习之旅&#xff0c;从数据集介绍、模型构建到…

U盘格式化了怎么办?这4个工具能帮你恢复数据。

如果你思维U盘被格式化了&#xff0c;也不用太过担心&#xff0c;其实里面的数据并没有被删除&#xff0c;只是被标记为了可覆盖的状态。只要我们及时采取正确的数据恢复措施&#xff0c;就有很大的机会可以将数据找回。比如使用专业得的数据恢复软件&#xff0c;我也可以跟大家…

Keysight 下载信源 Visa 指令

用于传输原始的IQ数据 file.wiq 或者 file.bin wave_bin:bytes with open("./WaveForm.wfm","rb") as f:wave_bin f.read()log.info("File:WaveForm.wfm Size:%d Bytes"%len(wave_bin)) IMPL.sendCommand(":MEM:DATA \"WFM1:FILE1\&q…

使用 IntelliJ IDEA 连接到达梦数据库(DM)

前言 达梦数据库是一款国产的关系型数据库管理系统&#xff0c;因其高性能和稳定性而被广泛应用于政府、金融等多个领域。本文将详细介绍如何在 IntelliJ IDEA 中配置并连接到达梦数据库。 准备工作 获取达梦JDBC驱动&#xff1a; 访问达梦在线服务平台网站或通过其他官方渠道…

远程升级又双叒叕失败?背后原因竟然是。。。

最近又遇到了远程升级接连失败的情况&#xff0c;耐心和信心都备受折磨&#xff01; 事情是这样的&#xff1a;有客户反馈在乡村里频繁出现掉线的情况&#xff0c;不敢耽搁&#xff0c;赶紧联系小伙伴排查测试&#xff0c;最后发现&#xff0c;只有去年某一批模块在当下环境才…