vue检测数据变化的原理

vue监测数据变化的原理

vue会监视data中所有层次的数据。

监测对象类型的数据

原理

vue监测对象类型的数据通过setter实现,且要在new Vue时就传入要监测的数据。

对象中后追加的属性,Vue默认不做响应式处理;如需后续添加的属性做响应式,请使用如下API: Vue.set(target,propertyName/index,value)vm.$set(target,propertyName/index,value)

图例分析

image-20230716210922128

在第一步加工(数据劫持)中,vue给每一个数组(由data对象中所有的属性而形成一个数组)属性都添加了get和set方法,这是实现检测数据变化的关键。

待补充…

Vue.set()方法

为什么需要这个方法?

直接修改没有被Vue监控的内容不会引起页面重新渲染。

当data中的数据已经定好之后,想再往里面添加属性就不会这么容易了 (在开发人员的角度,在代码中加是可以的;但是是在用户角度,那么就需要考虑不在代码层面去操作,而是把代码中写好程序实现在页面操作添加属性这一功能) ;

vue实例中的属性是从 _data 中通过数据代理生成的,而 _data中属性的方法是在生成vue实例之前加工中赋予的;

后面如果再加属性;不能直接把属性添加到实例根部(这样会导致_data中没有这个属性以及属性的get和set方法,因为是直接添加到实例根部了,没有经过_data )

image-20230715171535398

也不能直接把属性直接添加到 vm._data.xx=‘xxxx’;这样也是行不通的,因为这样添加出来的属性没有get和set方法,因为这两个方法是在vm实例生成之前做的加工中添加的。

image-20230715172014873

用法

通过Vue.set或者vm.$set我们可以给对象加入监视属性,从而达到修改数据重新渲染页面的效果

vue.set( )

image-20230715173741578

vm.$set( )

image-20230715173353664

实际应用

原理:点击按钮触发函数,在函数中使用vue.set对student添加‘sex‘属性,此时Vue会为其添加监视属性,并通过setter重新渲染页面

<!DOCTYPE html>
<html>
	<head>
		<meta charset="UTF-8" />
		<title>Vue监测数据改变的原理</title>
		<!-- 引入Vue -->
		<script type="text/javascript" src="../js/vue.js"></script>
	</head>
	<body>
		<!-- 准备好一个容器-->
		<div id="root">
			<button @click="addSex">添加一个性别属性,默认值是男</button>
			<h2>姓名:{{student.name}}</h2>
			<h2 v-if="student.sex">性别:{{student.sex}}</h2>
		</div>
	</body>

	<script type="text/javascript">

		const vm = new Vue({
			el:'#root',
			data:{
				student:{
					name:'tom',
					age:{
						rAge:40,
						sAge:29,
					},
					friends:[
						{name:'jerry',age:35},
						{name:'tony',age:36}
					]
				}
			},
			methods: {
				addSex(){
					// Vue.set(this.student,'sex','男')
					this.$set(this.student,'sex','男')
				}
			}
		})
	</script>
</html>

监测数组类型的数据

通过包裹数组更新元素的方法实现,本质就是做了两件事:调用原生对应的方法对数组进行更新以及重新解析模板,进而更新页面。不能像监测对象那样修改。

这些原生的方法是 Array.prototype 原形上的方法,不是Array上的。

在Vue修改数组中的某个元素一定要用如下方法:

  • 使用这些API:push()、pop()、shift()、unshift()、splice()、sort()、reverse()
  • Vue.set() 或 vm.$set();用法同对象

特别注意:Vue.set() 和 vm.$set() 不能给vm 或 vm的根数据对象 添加属性!!!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Vue数据监测的原理是利用了JavaScript的Object.defineProperty()方法,通过对数据对象进行劫持,对其属性的读取和修改进行监听和拦截,从而实现数据响应式的效果。当数据发生变化时,Vue会自动检测变化并通知相关的视图进行更新,从而实现数据和视图的自动同步。同时,Vue还提供了一些API,如$watch和computed等,方便开发者对数据变化进行手动监听和处理。 ### 回答2: Vue数据监测的原理是通过使用Object.defineProperty()方法对数据对象进行劫持和监听,实现对数据变化进行监测。在Vue的实例化过程中,会遍历所有的数据对象,将每个属性转化为getter和setter来实现数据的劫持。 当数据被访问时,会触发getter函数,在getter函数中可以进行依赖收集,将当前正在执行的Watcher对象添加到依赖列表中。当数据被修改时,会触发setter函数,setter函数会通知依赖列表中的Watcher对象更新视图。 在getter函数中进行依赖收集的关键是通过Dep(依赖收集器)来实现的。每个属性对应一个Dep对象,在getter函数中会通过Dep.target属性存储当前的Watcher对象,然后将该Watcher对象添加到Dep对象的依赖列表中。而Watcher对象是在Vue的编译阶段创建的,一个Watcher对象实际上包含了一个用户定义的回调函数(用于更新视图)以及关联的组件实例。 在数据对象被修改时,setter函数会被触发,这时会通知Dep对象的notify()方法去通知所有的Watcher对象进行更新操作。在更新过程中,Watcher会重新执行页面渲染的操作,将修改后的数据更新到视图上。 总结来说,Vue数据监测的原理是通过Object.defineProperty()方法将数据对象的属性转化为getter和setter,在getter函数中进行依赖收集,在setter函数中通知依赖进行更新。这样就实现了数据和视图的双向绑定,使得数据变化能够自动更新到对应的视图上。 ### 回答3: Vue数据监测原理是通过使用数据劫持结合发布-订阅模式来实现的。 首先,在Vue创建实例时,所有的data属性会被Vue转化成getter和setter的形式,并使用Object.defineProperty()进行劫持。 接下来,当访问或修改data属性时,Vue会收集相关依赖。当属性被读取时,会触发getter函数,将依赖收集到当前的依赖列表中;当属性被修改时,会触发setter函数,通知依赖更新。 然后,Vue利用发布-订阅模式建立了一个观察者(Watcher)与数据(data)之间的依赖关系。当依赖发生变化时,会通知相应的观察者进行更新操作。 另外,为了减少不必要的依赖收集和更新操作,Vue还进行了一些优化。Vue使用异步更新策略将多次数据变化的更新合并为一次更新,避免频繁的DOM操作。同时,还对监听的数据进行了缓存,当多次访问同一个属性时,只会收集依赖一次。 总结起来,Vue数据监测原理是通过数据劫持结合发布-订阅模式实现的。通过定义getter和setter来劫持属性,收集依赖并建立观察者与数据之间的依赖关系。当依赖发生变化时,触发相应的更新操作。同时,还进行了一些优化措施来提高性能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值