Vue 入门到实战 八

embedded/2025/2/9 0:28:25/

8章 组合API与响应性

目录

8.1 响应性

8.1.1 什么是响应性

8.1.2 响应性原理

8.2 为什么使用组合API

8.3 setup组件选项

8.3.1 setup函数的参数

8.3.2 setup函数的返回值

8.3.3 使用ref创建响应式引用

8.3.4 setup内部调用生命周期钩子函数

8.4 提供/注入

8.4.1 provide方法

8.4.2 inject方法

8.5 模板引用

8.6 响应式计算与侦听

8.6.1 响应式计算

8.6.2 响应式侦听


8.1 响应性

8.1.1 什么是响应性

响应性是一种允许我们以声明式的方式去适应变化的一种编程范例。Vue.js如何追踪数据的变化呢?在生成Vue.js实例时,使用带有gettersetter的处理程序遍历传入的data,将其所有property转换为Proxy对象。Proxy代理对象,顾名思义,在访问对象前增加一个中间层,通过中间层做一个中转,通过操作代理对象,实现目标对象的修改。Proxy对象对于用户来说是不可见的,但在内部,它使Vue.js能够在property值被访问或修改的情况下进行依赖跟踪和变更通知。

8-1】property转换为Proxy对象。

javascript"><script>const data = {uname: 'chenheng',age: 90}const handler = {get(target, name, receiver) {alert('执行get方法')//Reflect.get 方法查找并返回 target 对象的 name 属性,如果没有该属性,则返回 undefinedreturn Reflect.get(...arguments)},set(target, name, value, receiver) {alert('执行set方法')//Reflect.set 方法设置 target 对象的 name 属性等于 value。return Reflect.set(...arguments)}}const proxy = new Proxy(data, handler)alert(proxy.uname)    //执行get方法proxy.uname = 'hhhhh' //执行set方法alert(proxy.uname)    //执行get方法
</script>

target

要包装的目标对象Proxy。它可以是任何类型的对象,包括本机数组,函数甚至其他代理

handler

一个对象,其属性是定义对代理p执行操作时的行为的函数

proxy监听数组

 proxy可以监听属性的新增删除操作

proxy监听深层次嵌套对象

8.1.2 响应性原理

reactive()方法和watchEffect()方法是Vue3中响应式的两个核心方法,reactive()方法负责将数据变成响应式代理对象,watchEffect()方法的作用是监听数据变化去更新视图或调用函数。

8-2】reactive()方法和watchEffect()方法的应用。

8.2 为什么使用组合API

通过创建Vue.js组件,可以将接口的可重复部分及其功能提取到可重用的代码段中,从而使应用程序可维护且灵活。然而,当应用程序非常复杂(成百上千组件)时,再使用组件的选项(datacomputedmethodswatch)组织逻辑,可能导致组件难以阅读和理解。如果能够将与同一个逻辑相关的代码配置在一起将有效解决逻辑复杂、可读性差等问题。这正是使用组合API的目的。

8.3 setup组件选项

Vue组件提供setup选项,供开发者使用组合APIsetup选项在创建组件前执行,一旦props被解析,便充当组合式API的入口点。由于在执行setup时尚未创建组件实例,因此在setup选项中没有this。这意味着,除了props之外,无法访问组件中声明的任何属性,包括本地状态、计算属性或方法。

setup选项是一个接受propscontext参数的函数。此外,从setup返回的所有内容都将暴露给组件的其余部分(计算属性、方法、生命周期钩子、模板等等)。

8.3.1 setup函数的参数

1setup函数中的第一个参数(props

setup函数中的props是响应式的,当传入新的属性时,它将被更新。

8-3】setup函数中,参数props是响应式的。

但是,因为props是响应式的,不能使用ES6解构,将会消除props的响应性。如果需要解构props,可以在setup函数中使用toRefs函数来完成此操作。

8-4】setup函数中,使用toRefs函数创建props属性的响应式引用。

toRef是把对象的某个属性改成响应式的数据,toRefs是把整个对象改成响应式数据

2setup函数中的第二个参数(context

context上下文是一个普通的JavaScript对象,它暴露组件的4个属性:attrsslotsemit以及expose

setup(props, context) {

    // Attribute (非响应式对象,等同于 $attrs)

    console.log(context.attrs)可以获取父组件的传递归来的参数hobby,但是一定需要注释props

    // 插槽 (非响应式对象,等同于 $slots)

    console.log(context.slots)

    // 触发事件 (方法,等同于 $emit)

 

 home.vue

javascript"><template><Demo @zemit="showemit"></Demo>
</template><script>
import Demo from "@/components/demo.vue";
export default {components: {Demo,},setup() {function showemit(val) {alert(`触发context.emit事件,收到参数是:${val}`);}return {showemit,};},
};
</script>

 demo.vue

javascript"><template><p>个人信息</p><p>姓名:{{ person.name }}</p><p>年龄:{{ person.age }}</p><button @click="zevent">zemit事件</button>
</template><script>
import { reactive } from "vue";
export default {name: "Home12",emits: ["zemit"],setup(props, context) {console.log("1", props);// console.log("context.attrs", context.attrs);console.log("context.attrs", context.emit);const person = reactive({name: "刘巍",age: 18,});function zevent() {context.emit("zemit", '南昌大学');}return {  person,zevent,};},
};
</script>

 

8.3.2 setup函数的返回值

1)对象

如果setup返回一个对象,则可以在组件的模板中访问该对象的属性。

8-5】在该实例中,setup函数返回一个对象。

javascript"><template><h1>一个人的信息</h1><h3>职业:{{ job.type }}</h3><h3>薪水:{{ job.salary }}</h3><h3>爱好:{{ hobby }}</h3><h3>测试数据的值:{{ job.a.b.c }}</h3><button @click="changeInfo">修改人的信息</button>
</template><script>
import {reactive} from 'vue'export default {name: 'App',setup() {//数据let job = reactive({type: 'SAP工程师',salary: '60k',a: {b: {c: 666}}})let hobby = reactive(['篮球', '说泡', '旅游'])//counts changeInfo = ()=>{...}function changeInfo() {job.type = "管理咨询顾问"job.salary = "100k"job.a.b.c = 999hobby[0] = '学习'}return {job,hobby,changeInfo}}
}
</script>

2)渲染函数

setup还可以返回一个渲染函数,该函数可以直接使用在同一作用域中声明的响应式状态。

8-6】实现8-5】的功能,要求setup返回渲染函数。

8.3.3 使用ref创建响应式引用

1.声明响应式状态

要为JavaScript对象创建响应式状态,可以使用reactive()方法。reactive()方法接收一个普通对象然后返回该对象的响应式代理。示例代码如下:

const book = Vue.reactive({ title: '好书' })

reactive()方法响应式转换是“深层的”即影响对象内部所有嵌套的属性。基于ESProxy实现,返回的代理对象不等于原始对象。建议使用代理对象,避免依赖原始对象。

2.使用ref创建独立的响应式值对象

ref接受一个参数值并返回一个响应式且可改变的ref对象。ref对象拥有一个指向内部值的单一属性.value。示例代码如下:

const readersNumber = Vue.ref(1000)

console.log(readersNumber.value) //1000

readersNumber.value++

console.log(readersNumber.value) // 1001

ref作为渲染上下文的属性返回(即在setup()返回的对象中)并在模板中使用时,它会自动开箱,无需在模板内额外书写.value

8.3.4 setup内部调用生命周期钩子函数

setup内部,可通过在生命周期钩子函数前面加上“on来访问组件的生命周期钩子函数。因为setup是围绕beforeCreatecreated生命周期钩子函数运行的,所以不需要显式地定义它们。换句话说,在这些钩子函数中编写的任何代码都应该直接在setup函数中编写。这些on函数接受一个回调函数,当钩子函数被组件调用时将会被执行。示例代码如下:

setup() {

 // mounted时执行

 onMounted(() => {

  console.log('Component is mounted!')

  })

}

8.4 提供/注入

通过4.3.4节可知,使用provideinject可实现组件链传值。也就是说,父组件可以作为其所有子组件的依赖项提供程序,而不管组件层次结构有多深,父组件有一个provide选项来提供数据,子组件有一个inject选项来使用这个数据。(跨组件的数据传递)

现在,在组合API中,也可以使用provide方法和inject方法实现传值,但两者都只能在当前活动实例的setup()期间调用。

8.4.1 provide方法

首先,从vue显式导入provide方法;然后,在setup()中使用provide方法定义每个property

provide方法有两个参数:

l  name:代表字符串类型的属性名称;

l  value:代表任意类型的属性值。

8.4.2 inject方法

首先,从vue显式导入inject方法;然后,在setup()中使用inject方法注入每个property值。

inject方法有两个参数:

l  name:被注入的属性名称(字符串类型);

l  defaultValue:默认值(可选)。

假设我们有一个祖先组件 app,一个中间组件 one,以及一个后代组件 two。我们希望从 app传递一个数据到 two

app.vue

javascript"><template><one></one>
</template><script setup>
// import HelloWorld from './components/HelloWorld.vue'
// import setup from './components/setup.vue'
// import setup1 from './components/setup1.vue'
import one from './components/one.vue'
import { provide } from 'vue';
const hcm = 'payroll';
provide('sap', hcm)
</script><style>
#app {font-family: Avenir, Helvetica, Arial, sans-serif;-webkit-font-smoothing: antialiased;-moz-osx-font-smoothing: grayscale;text-align: center;color: #2c3e50;margin-top: 60px;
}
</style>

 one.vue

javascript"><template><two/>
</template><script setup>
import two from '../components/two.vue';
</script>

two.vue

javascript"><template><div :class="theme">我是SAP {{ theme }} 顾问.</div>
</template><script setup>
import { inject } from 'vue';const theme = inject('sap', 'time'); // 'time' 是默认值,如果 provide 没有提供 'theme'
</script><style scoped>
.dark {background-color: black;color: white;
}.light {background-color: white;color: black;
}
</style>

8.5 模板引用

在使用组合API时,响应式引用和模板引用的概念是统一的。为了获得对模板内元素或组件实例的引用,可以声明一个ref并从setup()返回。

8-8】在模板中使用ref引用响应式对象。

8.6 响应式计算与侦听

8.6.1 响应式计算

使用响应式计算方法computed有两种方式:传入一个getter函数,返回一个默认不可手动修改的ref对象;传入一个拥有getset函数的对象,创建一个可手动修改的计算状态。

8-9】返回一个默认不可手动修改的ref对象。

javascript"><template><div>{{count}}</div>
</template><script>
import{ref,computed} from 'vue';export default {setup() {const count = ref(1)const account = computed(()=> count.value + 1)console.log(account.value)account.value++// 返回值会暴露给模板和其他的选项式 API 钩子return {count}}
}
</script>

8-10】返回一个可手动修改的ref对象。

javascript"><template><div>{{count}}</div>
</template><script>
import{ref,computed} from 'vue';export default {setup() {const count = ref(1)const account = computed({get:()=> count.value + 1,set:(val)=> {count.value = val - 1},} )account.value = 1console.log(count.value)return {count}}
}
</script>

8.6.2 响应式侦听

可使用响应性侦听watchEffect方法,对响应性进行侦听。该方法立即执行传入的一个函数,同时响应式追踪其依赖,并在其依赖变更时重新运行该函数。(监听所有属性)

8-11】响应性侦听watchEffect方法的使用。

监听属性变化

javascript"><template><div><input type="text" v-model="obj.name"> </div>
</template><script>
import{reactive,watchEffect} from 'vue';export default {setup() {let obj = reactive({name:'vivi'});watchEffect(()=>{console.log('name:',obj.name)})return {obj}}
}
</script>

停止监听

javascript"><template><div><input type="text" v-model="obj.name"> <button @click="stopWatchEffect">停止监听</button></div>
</template><script>
import{reactive,watchEffect} from 'vue';export default {setup() {let obj = reactive({name:'vivi'});const stop1 =  watchEffect(()=>{console.log('name',obj.name)})const stopWatchEffect = ()=>{console.log('停止监听')stop1(); // ...当该侦听器不再需要时}return {obj,stopWatchEffect,}}
}
</script>

副作用

使用 onInvalidate 清理计时器,每次 count 变化时,watchEffect 会重新执行,在此之前 onInvalidate 会先清理掉之前的计时器,避免重复创建计时器导致内存泄漏。

注意:如果在 watchEffect 没有直接使用 count.value ,那么它的变化就不会触发副作用函数重新执行,从而不会调用 onInvalidate 清理之前的计时器

javascript"><template><div><p>当前计数: {{ count }}</p><button @click="count++">增加计数</button></div>
</template>
<script>
import { ref, watchEffect } from 'vue';
export default {setup() {const count = ref(0);watchEffect((onInvalidate) => {// 在函数内直接读取 count.value,确保它被追踪,这一步很重要!!!console.log(`副作用函数执行,count 值为: ${count.value}`);const timer = setInterval(() => {console.log(`计时器中 count 的值: ${count.value}`);}, 1000);onInvalidate(() => {console.log('清除计时器 timer');clearInterval(timer);});});return {count,};},
};
</script>


http://www.ppmy.cn/embedded/160666.html

相关文章

【C语言】指针详细解读3

1. 数组名的理解 我们使用指针一般访问数组内容时&#xff0c;我们可能会这样写&#xff1a; int arr[10] {1,2,3,4,5,6,7,8,9,10}; int *p &arr[0]; 这⾥我们使⽤ &arr[0] 的⽅式拿到了数组第⼀个元素的地址&#xff0c;但是其实数组名本来就是地址&#xff0c;⽽…

组合总和(力扣39)

这道题又在之前的基础上进行了变形。递归是在一个集合里进行&#xff0c;但每次递归我们可以选择重复的数字&#xff0c;这代表递归时不需要缩小集合范围。但是组合的无序性仍要考虑&#xff0c;所以每一层for循环的起始值还是需要用变量控制。另外&#xff0c;我们可以事先对元…

基于深度学习的医疗器械分类编码映射系统:设计、实现与优化

一、引言 1.1 研究内容与方法 本研究旨在设计并实现一个基于深度学习的医疗器械分类编码映射系统,以解决医疗器械分类编码管理中存在的问题,提高医疗器械管理的效率和准确性。具体研究内容包括以下几个方面: 系统需求分析:深入研究国内外主流医疗器械编码体系,如中国的 …

使用媒体查询确保网页能够在手机、平板和电脑上正常浏览

为了确保网页能够在手机、平板和电脑上正常浏览&#xff0c;媒体查询的设置需要考虑到不同设备的屏幕尺寸和分辨率。以下是一些建议的像素设置范围&#xff1a; 手机 宽度设置&#xff1a;手机设备的网页宽度通常可以设置在320像素到480像素之间。这个范围可以覆盖大部分主流…

Redis Copilot:基于Redis为AI打造的副驾工具

我们最近发布了Redis Copilot&#xff0c;以帮助开发者更快地使用Redis构建应用。我们的使命是使应用程序快速运行&#xff0c;并简化构建过程。为此&#xff0c;Redis Copilot作为您的AI助手&#xff0c;能够让您更迅速地完成与Redis相关的任务。您今天就可以在Redis Insight中…

SQL Server2019下载及安装教程

一、软件下载 SQLServer2019及SSMS管理工具下载链接&#xff1a; 百度网盘 请输入提取码 二、SQLServer2019安装 选中要安装的iso映像文件&#xff0c;右键点击装载&#xff08;有些系统可以直接双击打开&#xff0c;有些需要安装Daemon Tools软件去打开&#xff09; 找到s…

【RK3588嵌入式图形编程】-SDL2-创建应用事件循环

创建应用事件循环 文章目录 创建应用事件循环1、概述2、事件队列与事件循环3、创建应用循环和事件循环4、处理错误5、退出应用程序6、总结在本文中,将详细介绍应用程序的事件循环。 1、概述 实现任何应用程序的第一步是编写保持程序运行的基础代码,直到我们或用户决定退出。…

使用 CMake 自动管理 C/C++ 项目

使用 CMake 自动管理 C/C 项目 1. 介绍 CMake 是一个强大的构建系统&#xff0c;可用于跨平台管理 C/C 项目的编译过程。本 CMakeLists.txt 文件提供了一种自动化的方式来管理 C/C 项目&#xff0c;包括创建代码目录、自动编译所有源文件、管理输出文件等。 2. CMake 最低版…