当前位置: 首页 > news >正文

vue3 分析总结响应式丢失问题原因(二)

上一篇文件理解了响应式对象应用原理了。公式:

响应式对象 = 代理 + 触发器。

真实的对象:指的是就是一个普通的Object对象,类似{a: 1}

代理对象:Vue定义的特殊对象。类似ref、reactive、computed、props等对象。

但是实际使用结果和预期还是不一致。具体现象是数据修改了,但是并没有实现响应式更新界面。即出现了响应式丢失现象。

一、什么情况下对象的响应式会丢失?

一般网络上查资料,都是告诉我们常见的某些操作(如解构、替换)中容易丢失响应性。只是笼统的说一下容易丢失,具体为什么丢失,会举例哪些情况会丢失,说一堆,感觉把所有情况都告诉你了,又感觉好像还缺点什么。我总结了一下,好像就是缺了一点抽象概况的总结。

所以我先说结论:造成响应式丢失的原因就是没有正确的获取到代理对象。

替换对象时丢失

首先比较简单一点,就是替换操作。例子:

let state = reactive({ count: 0 });
state = { count: 1 }; // 新对象没有响应性

进行替换操作时,可能新的值是一个实际的对象,而不是响应式对象,导致变量引用对象发送了改变,实际结果就是state响应性丢失了。这种情况比较好理解的,注意一点就好了。

解构对象时丢失

比如容易出问题的地方是解构操作。后端程序员可能对于解构概念不是很清晰,理解其语法糖本质后,后面就好理解了。

解构的基本概念

解构是一种从数组或对象中提取值的语法。它允许你将数组或对象的属性直接赋值给变量。

普通数组解构 和 等价的js代码

const arr = [1, 2, 3];
const [a, b, c] = arr; // 解构数组
console.log(a); // 1
console.log(b); // 2
console.log(c); // 3
const arr = [1, 2, 3];
const a = arr[0]; 
const b = arr[1]; 
const c = arr[2]; 
console.log(a); // 1
console.log(b); // 2
console.log(c); // 3

普通对象解构 和 等价的js代码

const obj = { name: "Alice", age: 25 };
const { name, age } = obj; // 解构对象
console.log(name); // "Alice"
console.log(age); // 25
const obj = { name: "Alice", age: 25 };
const name = obj.name; 
const age = obj.name; 
console.log(name); // "Alice"
console.log(age); // 25

阶段总结,解构就是一个语法糖,和普通js赋值操作没什么区别。 

reactive对象解构

这个要分为两种情况:

第一种,如果解构的属性值是原始值类型,那么返回的值就是一个真实对象,而不是代理对象,即返回的值丢失了响应性。

在 JavaScript 中,原始值(Primitive Values) 是最基本的数据类型,它们不是对象,也没有方法或属性。JavaScript 有以下 7 种原始值数据类型:

number(数字)、string(字符串)、boolean(布尔值)、undefined(未定义)、null(空值)、bigint(大整数)、symbol(符号)。

一般我们最常见的就是数字和字符串了。

第二种,如果解构的属性值是原始值类型,那么返回的值就是reactive对象,即返回的值保留了响应性。

ref对象解构

这个同样要分为两种情况:

第一种,如果ref代理的实际对象值是原始值类型,那么返回的值就是一个真实对象,而不是代理对象,即返回的值丢失了响应性。

const count = ref(0); //ref对象
const { value: myCount } = count; // 解构ref
console.log(myCount); // 普通对象,丢失了响应性。

第二种,如果ref代理的实际对象值是原始值类型,那么返回的值就是reactive对象,即返回的值保留了响应性。此时如果继续解构,则参考上面的reactive对象解构。

  const aaa = ref({a:{b:"a"}});
  const { value } = aaa; //ref对象解构
  const { a } = value;//reactive对象解构
  const { b } = a;//reactive对象解构

  console.log("aaa", aaa); //ref对象
  console.log("value", value);//reactive对象
  console.log("a", a);//reactive对象
  console.log("b", b);// 普通对象(无响应性)

  console.log("value.a", value.a);//reactive对象
  console.log("value.a.b", value.a.b);// 普通对象(无响应性)

总结,解构时是否会丢失响应性主要看解构的返回值是否是原始值类型,是的话就会丢失响应性,否则会保留响应性。并且解构返回值要么是原始值类型,要么是reactive对象(即ref解构后也是返回reactive对象)。

换一个说法,因为解构就是一个语法糖,和普通js赋值操作没什么区别,ref对象和reactive对象读取数据返回值 与 解构的返回值 逻辑是相同的,所以我们可以得出以下结论:

  1. ref.value 读取的结果要么是原始值类型,要么是reactive对象。
  2. ref.value 或者 reactive.attr 返回结果要么是原始值类型,要么是reactive对象。

二、常见响应式应用情景分析 

computed

computed 可以看做是一个只读ref对象,即ref.value值不能修改。还有一点就是computed.value返回的是一个真实的值,而不是响应式对象。

    const test1 = computed(()=>{
		return  {a:1};
	});
	const test2 = ref({b:2});
	console.log(test1.value); //结果是一个真实的对象{a:1},即结果没有响应性,只是一个普通对象
	console.log(test2.value);//结果是一个代理的对象{b:1}, 即结果还是有响应性

ref与reactive函数创建的对象是否是单例模式。

结论:ref创建的对象不是单例模式,reactive创建的对象是单例模式。

测试代码:

	const testObj = {a:"1"};
	const testObjRef1 = ref(testObj);
	const testObjRef2 = ref(testObj);
	const testObjReactive3 = reactive(testObj);
	const testObjReactive4 = reactive(testObj);
	console.log("testObjRef1.value == testObjReactive3", testObjRef1.value == testObjReactive3);
	console.log("testObjRef1 == testObjRef2", testObjRef1 == testObjRef2);
	console.log("testObjRef1.value == testObjRef2.value", testObjRef1.value == testObjRef2.value);
	console.log("testObjReactive4 == testObjReactive4", testObjReactive4 == testObjReactive4);

运行结果:

 

为什么关注是否单例问题,是担心不同的响应式对象变量引用的代理对象不是同一个,那么代理对象绑定的触发器可能不一样,那响应式结果可能跟预期不一样。但实际测试发现reactive创建的对象是单例的,就是同一个对象创建响应式对象使用的是同一个代理对象。而ref创建的对象虽然不是单例,但是其值如果不是原始值类型,则ref.value 值是reactive对象,是单例的。

所以,可以简单的理解为相同对象在不同组件页面或者任何地方获取的响应性都是同一个响应性对象。

子组件的属性的响应式情况

props的响应式比较特殊。既不是ref对象,也不是reactive对象。而是单独实现的一套代理逻辑。其具体逻辑如下:

先看例子:

父组件:

    const testB = ref("1");
	console.log("testB" , testB)
	
	let a = {a:"a"};
	const aRef = ref(a);
	const aReactive = reactive(a);
	
	let bObj = "B";
	const bObjRef = ref(bObj);
	// const bReactive = reactive(bObj); //reactive不支持基本类型
	watch(testB, async (newAttr, oldAttr) => {
		// a = {a:newAttr};  //情况1:只有props.a会跟着变,有响应式渲染。
		// a.a = newAttr; //情况2:props.a、props.aRef、props.aReactive 对应的值会跟着变,但是不会没有响应式渲染。
		// bObj = newAttr; //情况3 只有props.bObj会跟着变,有响应式渲染。
		// bObjRef.value = newAttr;//情况4 只有props.bObjRef会跟着变,有响应式渲染。
        // aReactive.a = newAttr;//情况5 props.a、props.aRef、props.aReactive 对应的值会跟着变,有响应式渲染。
	})
	
<Component  :a="a" :aRef="aRef" :aReactive="aReactive" :b="b" :bRef="bRef"></Component>

 然后再Component子组件的script中查看接收的属性:

const props = defineProps({
	  a:Object,
	  aRef:Object,
	  aReactive:Object,
	  b:String,
	  bRef:String,
	});
	console.log("props.a", props.a);
	console.log("props.aRef", props.aRef);
	console.log("props.aReactive", props.aReactive);
	console.log("props.b", props.b);
	console.log("props.bRef", props.bRef);
<template>
	<div> props.a ->  {{props.a}} </div>
	<div> props.aRef ->  {{props.aRef}} </div>
	<div> props.aReactive ->  {{props.aReactive}} </div>
	<div> props.bObj ->  {{props.bObj}} </div>
	<div> props.bObjRef ->  {{props.bObjRef}} </div>
</template>

运行结果:

初步结论:

  1. props既不是ref对象,也不是reactive对象,是vue3中特殊的响应式对象。他的属性值是只读的,不能修改,即props.attr的值不能修改,他的值只能父组件中修改,修改后由于props是响应式对象,子组件中的props.attr值会跟着改变。
  2. 如果属性值是原始值类型,则通过props访问该属性返回一个非代理对象。参考props.b和props.bRef。
  3. 如果传入属性值不是原始值类型,并且不是ref和reactive对象,则通过props访问该属性返回的就是传入的值。参考props.a。
  4. 如果传入属性值不是原始值类型,并且是ref或者reactive对象,则通过props访问该属性返回的就是传入的值的reactive对象。参考props.aRef和props.aReactive。
  5. 父组件中修改a的值,子组件中props.a的属性值会着变,有响应性,会重新渲染刷新页面。情况1.
  6. 父组件中修改a的属性值,即修改a.a的值,子组件中props.a、props.aRef、props.aReactive 对应的值会跟着变,但是没有响应性,即不会重新渲染刷新页面。情况2.
  7. 父组件中修改b的值,子组件中只有props.b会跟着变,即props.b有响应性。情况3.
  8. 父组件中修改bRef的值,子组件中只有props.bRef会跟着变,即props.bRef有响应性。情况4
  9. 父组件中修改aReactive的属性值,即修改aReactive.a的值,子组件中props.a、props.aRef、props.aReactive 对应的值会跟着变,有响应性,会重新渲染刷新页面。情况5.

 进一步结论:

  1. props的属性都是有响应性的,即只要传入的属性值的引用(比如:a="a" 中 变量a的值就是一个引用,只有a的值改变才会触发,a.a改变不会触发)发生了改变,即会触发props的属性的响应性,更新对应的属性值。
  2. props.aRef和props.aReactive是响应式对象,通过他们修改对象属性,父组件中对应属性也会跟着改变,并且会重新渲染DOM。

pinia状态的响应式

先看代码:

import { useRootNodeStore } from '@/stores/rootNode';
console.log("useRootNodeStore", useRootNodeStore);//useRootNodeStore 是一个函数
const rootNodeStore = useRootNodeStore();
console.log("rootNodeStore", rootNodeStore);//返回一个reactive对象。

运行结果:

结论:useRootNodeStore是一个函数。useRootNodeStore()结果 就是等价与 reactive(状态定义最后return的对象);比如状态定义如下:

export const useCounterStore = defineStore('counter', () => {
  const count = ref(0)
  const doubleCount = computed(() => count.value * 2)
  function increment() {
    count.value++
  }

  return { count, doubleCount, increment }
})

useRootNodeStore() 返回结果等价于:

reactive({ count, doubleCount, increment })

组合式函数中的响应式

首先看下官方例子:

// mouse.js
import { ref, onMounted, onUnmounted } from 'vue'

// 按照惯例,组合式函数名以“use”开头
export function useMouse() {
  // 被组合式函数封装和管理的状态
  const x = ref(0)
  const y = ref(0)

  // 组合式函数可以随时更改其状态。
  function update(event) {
    x.value = event.pageX
    y.value = event.pageY
  }

  // 一个组合式函数也可以挂靠在所属组件的生命周期上
  // 来启动和卸载副作用
  onMounted(() => window.addEventListener('mousemove', update))
  onUnmounted(() => window.removeEventListener('mousemove', update))

  // 通过返回值暴露所管理的状态
  return { x, y }
}
<script setup>
import { useMouse } from './mouse.js'

const { x, y } = useMouse()
console.log("useMouse", useMouse);//结果是一个函数
console.log("useMouse()", useMouse());//结果是一个reactive对象
</script>

<template>Mouse position is at: {{ x }}, {{ y }}</template>

结论:useMouse是一个函数。useMouse()结果 就是等价与 reactive({x,y});

组合式函数与pinia状态的区别?

组合式函数主要是为了在多组件中复用,调用函数后,相关与函数体内容在对应组件setup中执行。唯一注意一点区别就是,属性定义不能再函数中定义。const props = defineProps({}) 上面代码不能出现在组合式函数内。

pinia状态管理可以看做一个特殊的组合式函数,pinia状态管理的函数体只有首次调用执行一次,后续在调用直接reactive(之前返回的结果对象)。

相关文章:

  • 【从0做项目】Java搜索引擎(3)
  • 使用 DeepSeek 生成商城流程图
  • Rasa学习笔记
  • 基于HAL库的按钮实验
  • 【个人开发】deepspeed+Llama-factory 本地数据多卡Lora微调
  • 【AI战略思考15】我对做自媒体视频博主的初步探索和一些思考
  • ubuntu下ollama/vllm两种方式在本地部署Deepseek-R1
  • 数据结构 堆和priority_queue
  • 跨平台数字内容整合策略:提升全域用户体验的关键路径
  • 一键终结环境配置难题:ServBay 1.9 革新 AI 模型本地部署体验
  • 使用 HTML CSS 和 JAVASCRIPT 的黑洞动画
  • 解决IDEA报错:java 找不到符号
  • 鸿蒙应用开发者基础
  • AI前端开发:驶向国际化职业发展快车道
  • 09 解决方案 - 开源机器人+具身智能+AI
  • 2010年上半年软件设计师考试上午真题的知识点整理(附真题及答案解析)
  • Ubuntu 下 nginx-1.24.0 源码分析 - ngx_localtime 函数
  • 3D数字化技术:重塑“人货场”,开启营销新纪元
  • 机器学习周报-文献阅读
  • 刷SQL总结
  • 中央纪委办公厅公开通报3起整治形式主义为基层减负典型问题
  • 在上海生活8年,13岁英国女孩把城市记忆写进歌里
  • 银川市市长信箱被指已读乱回,官方回应
  • 六朝文物草连空——丹阳句容南朝石刻考察纪
  • 关键词看中国经济“一季报”:稳,开局良好看信心
  • 京东美团开打,苦了商家?