最近使用vue watch时,在某些模块监听不到对象的改变,无法触发回调函数。
解决:
使用watch监听对象时,只能监听到该对象初始化时已存在的key值。
如下例监听user对象,在初始化时没有age属性,那在mounted中给user.age赋值后不会触发watch中的回调:
var app = new Vue({
el: '#app',
data: {
user: {
name: 'zhangsan'
}
},
mounted() {
var _this = this
setTimeout(() => {
_this.user.age = 10
},2000)
},
watch: {
user: {
handler (val) {
console.log('user update')
},
deep:true
}
}
})
若想监听到这个变化,需要给user初始化的age:
data: {
user: {
name: 'zhangsan',
age: 1
}
},
问题原因及vue watch原理:
这个问题其实是比较低级的错误,说明对vue的设计原理和理念还是理解太浅。watch的基本用法这里不再赘述,下面通过遇到的这个问题,探究一下watch对object类型进行监听的用法和原理。
通过官方文档我们知道,当监听的目标用对象进行配置时,共有三个配置属性:
watch: {
a:{
handler: function (val, oldVal) { /* ... */ },
immediate:true,
deep: true
}
}
首先,handler是vue中定义好的属性名,在用对象配置时,回调函数只能写在handler中。如下图源码所示,在creatWatcher时,会验证传入的配置项handler是否为一个纯对象类型,如果是对象类型,则会取handler.handler作为真正的回调函数;而下面的代码则是只传递方法名或方法实体的处理逻辑。
关于deep属性,大家都知道vue的绑定原理是建立在Object.defineProperty的set和get方法上的。给某个属性绑定set和get之后,只有改变该属性本身时,才会触发set和get的回调函数,而改变其内部属性时不会触发。所以在vue内部创建watcher时会有一个traverse方法,当配置deep为true时,调用traverse遍历其下每一个子属性。如下源代码:
而我们又知道vue只有在组件初始化时,会对data中数据进行set和get的绑定。这也是为什么在后续的业务逻辑中,给某个对象插入新属性时不会触发watch监听的原因。
immediate属性为true时,watcher创建后会立刻执行回调,这点不需要过多的介绍。
vue watch监听不到对象,探究 watch 原理的更多相关文章
-
设计模式(5): vue 不监听绑定的变量
概述 最近最近做项目的时候总会思考一些大的应用设计模式相关的问题,我把自己的思考记录下来,供以后开发时参考,相信对其他人也有用. 绑定变量 一般情况下,如果我们需要在组件中使用某个变量,会这么使用: ...
-
vue时时监听input输入框中 输入内容 写法
Vue input 监听 使用 v-on:input="change" 实现即可 App.vue <template> <div> <md-field ...
-
Vue 事件监听实现导航栏吸顶效果(页面滚动后定位)
Vue 事件监听实现导航栏吸顶效果(页面滚动后定位) Howie126313 关注 2017.11.19 15:05* 字数 100 阅读 3154评论 0喜欢 0 所说的吸顶效果就是在页面没有滑动之 ...
-
详解Vue 如何监听Array的变化
详解Vue 如何监听Array的变化:https://www.jb51.net/article/162584.htm
-
9.Vue.js 监听属性
本章节,我们将为大家介绍 Vue.js 监听属性 watch,我们可以通过 watch 来响应数据的变化. 以下实例通过使用 watch 实现计数器: <div id = "app&q ...
-
vue 监听变量或对象
注意:监听的对象必须已经在data中声明了 data: { a: 1, b: 2, c: 3, d: 4, e: { f: { g: 5 } } }, watch: { a: function (va ...
-
vue watch监听对象及对应值的变化
data:{ a:1, b:{ value:1, type:1, } }, watch:{ a(val, oldVal){//普通的watch监听 console.log("a: " ...
-
Echarts图标宽度变成100px,让图表宽度随着父元素自动适应,Vue实时监听宽度的变化,这可能是史上最好的解决方案!
最近工作中element后台管理使用Echarts图表,本后台项目分图表模式和列表模式,使用display控制显示隐藏,这样就引出了本文的问题. 问题1:Echarts图标宽度变成100px? 问题2 ...
-
Vue -- 数据监听
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
随机推荐
-
一个jQ版大图滚动
难得周末能休息,也是越发的代码难受,手就想敲点东西,这不闲着无聊敲了一个Jq版的大图滚动,不足之处大家批评指正: 运作环境win7,代码编辑器是:sublime; 我把源码复制了一下, <!do ...
-
Json解析实例
using System; using System.Collections.Generic; using System.Runtime.Serialization; using System.Win ...
-
sqlite数据库 select 查询带换行符数据
在sqlite 数据库中用 select 语句查询带 换行符的 数据信息 实现 SELECT * from questions_exec where title like '%'||x'0 ...
-
max(min)-device-width和max(min)-width的区别
在网页自适应设计中,max-device-width和max-width是不可缺少的两大CSS属性,但是可能大家在使用的选择上没有太多讲究,认为用其中一个即可.确实,如果没有特定要求,用任何一个都没有 ...
-
C++ Iterator迭代器介绍及Iterator迭代器用法代码举例
C++ Iterator迭代器介绍 迭代器可被用来访问一个容器类的所包函的全部元素,其行为像一个指针.举一个例子,你可用一个迭代器来实现对vector容器中所含元素的遍历.有这么几种迭代器如下: 迭代 ...
-
iOS程序崩溃相关的处理办法
一.bug追踪 1.捕获异常:Exception breakpoint 步骤: 2.终止调用:Symbolic breakpoint 步骤:前两步和一 基本是一样的,不截图了,只是在第二步选择的时候选 ...
-
Java课程设计—学生成绩管理系统(201521123004-林艺如)
1.团队课程设计博客 团队课程设计博客链接 2.个人负责模块或任务说明 ①.Menu Menu.jsp 在页面中给出提示,用HTML的 MenuTeacher.jsp 利用Menu.jsp进行具体化完 ...
-
Access数据库SQL注入(Access SQL Injection)
一.Microsoft Office Access数据库手工注入语句 1.参数后面加 ’ .and 1=1.and 1=2看返回状态判断是否存在注入点 2.参数后面加 and exists(sel ...
-
redis持久化详述
本来打算根据自己搜索的一些文章写些总结,后来发现了一篇好文,这里转载下,在自己博客里面记录下. 原文链接:https://www.cnblogs.com/kismetv/p/9137897.html ...
-
jpa持久化对象四种状态
自己理解,不完全正确,大致如下: 例:某实体类 Person(int id,string name,int age); id 为主键. 新建:new Person(), 并且未给 id 赋值 ...