Vue监测数据改变的原理

Vue监视数据的原理:

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

2.如何监测对象中的数据?(通过setter实现监视,且要在new Vue时就传入要监测的数据)

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

3.如何监测数组中的数据?(通过包裹数组更新元素的方法实现,本质就是做了两件事:)

  • 调用原生对应的方法对数组进行更新
  • 重新解析模板,进而更新页面

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

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

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

先引出一个问题,可以更新人员信息的代码示例:(奏效)
点击按钮更新人员信息

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
    <h2>人员列表</h2>
    <button @click="updatedMei">更新马冬梅的信息</button>
    <ul>
        <li v-for="(p,index) in persons" :key="p.id">
            {{p.name}} - {{p.age}} - {{p.sex}}
        </li>
    </ul>
</div>
<script>
    Vue.config.productionTip = false
    const vm = new Vue({
        el: "#app",
        data: {
            persons: [
                {id: '001', name: '马冬梅', age: 19, sex: '女'},
                {id: '002', name: '周冬雨', age: 20, sex: '女'},
                {id: '003', name: '周杰伦', age: 21, sex: '男'},
                {id: '004', name: '温兆伦', age: 22, sex: '男'}
            ]
        },
        methods: {
            updatedMei() {
                this.persons[0].name = '马老师'
                this.persons[0].age = 50
                this.persons[0].sex = '男'
            }
        }
    })
</script>
</body>
</html>

点击按钮更新不了的代码示例:(不奏效)

这个实实在在的是把数据改了,只不过,vue没有监测到

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
    <h2>人员列表</h2>
    <button @click="updatedMei">更新马冬梅的信息</button>
    <ul>
        <li v-for="(p,index) in persons" :key="p.id">
            {{p.name}} - {{p.age}} - {{p.sex}}
        </li>
    </ul>
</div>
<script>
    Vue.config.productionTip = false
    const vm = new Vue({
        el: "#app",
        data: {
            persons: [
                {id: '001', name: '马冬梅', age: 19, sex: '女'},
                {id: '002', name: '周冬雨', age: 20, sex: '女'},
                {id: '003', name: '周杰伦', age: 21, sex: '男'},
                {id: '004', name: '温兆伦', age: 22, sex: '男'}
            ]
        },
        methods: {
            updatedMei() {
                this.persons[0] =  {id: '001', name: '马老师', age: 50, sex: '男'}
            }
        }
    })
</script>
</body>
</html>

打开控制台看一下:

现在模拟一个数据监测:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<script>
    let data = {
        name: '尚硅谷',
        address: '北京'
    }

    //创建一个监视的实例对象,用于监视data中属性的变化
    const obs = new Observer(data)
    console.log(obs)

    //准备一个vm实例对象
    let vm = {}
    vm._data = data = obs

    function Observer(obj) {
        //汇总对象中所有的属性形成一个数组
        const keys = Object.keys(obj)
        keys.forEach((k) => {
            //this指代Observer这个对象
            Object.defineProperty(this, k, {
                get() {
                    return obj[k]
                },
                set(val) {
                    console.log('${k}被改了,我要去解析模板,生成虚拟DOM,...我要开始忙了!')
                    obj[k] = val
                }
            })
        })
    }
</script>
</body>
</html>

Vue数据监视:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <style>
        button {
            margin-top: 10px;
        }
    </style>
    <script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
    <h1>学生信息</h1>
    <button @click="student.age++">年龄+1岁</button>
    <br>
    <button @click="addSex">添加性别属性,默认值男</button>
    <br>
    <button @click="addFriend">在列表首位添加一个朋友</button>
    <br>
    <button @click="updateFirstFriendName">修改第一个朋友的名字为:张三</button>
    <br>
    <button @click="addHobby">添加一个爱好</button>
    <br>
    <button @click="updateHobby">修改第一个爱好为:开车</button>
    <br>
    <button @click="removeSmoke">过滤掉爱好中的抽烟</button>
    <br>
    <h3>姓名:{{student.name}}</h3>
    <h3>年龄:{{student.age}}</h3>
    <h3 v-if="student.sex">性别:{{student.sex}}</h3>
    <h3>爱好</h3>
    <ul>
        <li v-for="(h,index) in student.hobby" :key="index">
            {{h}}
        </li>
    </ul>
    <h3>朋友们</h3>
    <ul>
        <li v-for="(f,index) in student.friends" :key="index">
            {{f.name}} -- {{f.age}}
        </li>
    </ul>
</div>
<script>
    Vue.config.productionTip = false
    const vm = new Vue({
        el: "#app",
        data: {
            student: {
                name: 'tom',
                age: 18,
                hobby: ['抽烟', '喝酒', '烫头'],
                friends: [
                    {name: 'jerry', age: 35},
                    {name: 'tony', age: 36}
                ]
            }
        },
        methods: {
            addSex() {
                // Vue.set(this.student,'sex','男')
                this.$set(this.student, 'sex', '男')
            },
            addFriend() {
                this.student.friends.unshift({name: 'jack', age: 70})
            },
            updateFirstFriendName() {
                this.student.friends[0].name = '张三'
            },
            addHobby() {
                this.student.hobby.push('学习')
            },
            updateHobby() {
                this.student.hobby.splice(0, 1, '开车')
            },
            removeSmoke() {
                this.student.hobby = this.student.hobby.filter((h) => {
                    return h !== '抽烟'
                })
            }
        }
    })
</script>
</body>
</html>

  • 1
    点赞
  • 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
发出的红包

打赏作者

阿瞒有我良计15

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值