Vue数据双向绑定原理(vue2向vue3的过渡)

简介: 众所周知,Vue的两大重要概念:数据驱动组件系统接下来我们浅析数据双向绑定的原理

一、vue2


1、认识defineProperty


vue2中的双向绑定是基于definePropertyget操作set操作,那么我们简单认识下defineProperty作用: 就是直接在一个对象上定义一个新属性,或者修改一个已经存在的属性。 那么我们先来看下Object.getOwnPropertyDescriptor(),有定义方法就会有获取方法,对这就是与defineProperty相对的方法,它可以获取属性值。


var a ={
    b:1,
    c:2
}
console.log(Object.getOwnPropertyDescriptor(a,'b')); //查看获取b属性


网络异常,图片无法展示
|


这就是打印出来的结果,configurable的意思是可便利,enumerable的意思是可枚举,writable的意思是可写。


2、使用defineProperty实现简单的私有变量


知道了属性内部的属性值,我们可以使用defineProperty来实现一个私有变量。


var a ={
    b:1,
    c:2
}
Object.defineProperty(a,'b',{
    writable:false  //不可写
})
console.log(Object.getOwnPropertyDescriptor(a,'b')); //查看获取b属性


网络异常,图片无法展示
|


发现,改变不了a.b的值。


另外,还可以使用一个很快捷的方法实现私有变量


var a ={
    b:1,
    c:2
}
Object.freeze(a,'b'); //冻结,可理解为变为私有属性
// Object.seal(a,'b'); //configurable置为false 
console.log(Object.getOwnPropertyDescriptor(a,'b')); //查看获取b属性


网络异常,图片无法展示
|


同样,writable为false。如果使用seal()只能使configurable为false。


3、简单认识defineProperty的get、set的方法


我们先不要创建全局变量,跟get方法return值,你会发现值为undefined


网络异常,图片无法展示
|


于是我们改了改


var a ={
    b:1,
    c:2
}
 var _value=a.b; //b值赋值需要先赋给全局变量,不然使用不了
 Object.defineProperty(a,'b',{
    get:function(){
        console.log('get');
        return _value; // get方法必须ruturn值,值为全局变量。
    },
    set:function(newValue){
        console.log(newValue)
        _value=newValue; // 赋值
    }
 })


网络异常,图片无法展示
|


取到了。

4、使用defineProperty简单实现Vue双向绑定


test.js


// 使用defineProperty
function vue () {
    this.$data={
        a:1 // 数组的话不会更新
    };
    this.el=document.getElementById('app');
    this._html="";
    this.observe(this.$data);
    this.render();
};
vue.prototype.observe=function(obj){
    var value;
    var self=this;
    for(var key in obj){
        value =obj[key];
        if(typeof value === 'object'){
            this.observe(value)
        }else{
            Object.defineProperty(this.$data,key,{
                get:function(){
                    return value;
                },
                set:function(newvalue){
                    value=newvalue;
                    self.render();
                }
            })
        }
    }
}
vue.prototype.render=function(){
    this._html='I am '+ this.$data.a;
    this.el.innerHTML=this._html;
}
// ***************************************
//数组改变更新,装饰者模式
// var arraypro=Array.prototype;
// var arrayob=Object.create(arraypro);
// var arr = ["push","pop","shift"];
// // arr里的方法,既能保持原有的方法,又能触发更新。
// arr.forEach(function(method,index){
//     arrayob[method]=function(){
//         var ret=arraypro[method].apply(this,arguments);
//         console.log('更新');
//         return ret;
//     }
// })
// var arr=[];
// arr.__proto__=arrayob;
// arr.push(1);


test.html


<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>vue双向绑定原理</title>
</head>
<body>
    <div id="app"></div>
</body>
<script src="test.js"></script>
<script>
    var vm = new vue();
    setTimeout(function(){
        console.log('changes');
        console.log(vm.$data);
        vm.$data.a=222;
    },2000)
</script>
</html>


这样我们就简单实现了数据双向绑定。


二、vue3


1、简单认识Proxy


就是一种机制,用来拦截外界对目标对象的访问,可以对这些访问进行过滤或者改写,所以Proxy更像是目标对象的代理器。


var ob= {
    a:1,
    b:2
}
// 新创建一个Proxy对象直接替换ob对象
ob=new Proxy(ob,{
    get:function(target,key,receive){   // target指的是原始对象,key指的是属性值,receive指的是这个Proxy对象
        console.log(target,key,receive) 
        return target[key]; //get方法 依然需要return
    },
    set:function(target,key,newvalue,receive){
        console.log(target,key,newvalue,receive) // newvalue指新创建的值
        target[key]=newvalue;
    }
});


这里需要注意的是:第一次使用proxy时,在new的时候把原对象替换掉。就会触发get方法


看到上方的代码,我们可以总结Proxy的几个优点:


  1. defineProperty只能监听某个属性,不能对全对象监听;
  2. 所以可以省去for in 提升效率;
  3. 可以监听数组,不用再去单独的对数组做异性操作。


2、使用Proxy简单实现数据双向绑定


test1.js


// 使用Proxy
function vue () {
    this.$data={
        a:1 
    };
    this.el=document.getElementById('app');
    this._html="";
    this.observe(this.$data);
    this.render();
};
vue.prototype.observe=function(obj){
    var self=this;
    this.$data=new Proxy(this.$data,{
        get:function(target,key){
            return target[key];
        },
        set:function(target,key,newvalue){
            target[key]=newvalue;
            self.render();
        }
    })
}
vue.prototype.render=function(){
    this._html='I am '+ this.$data.a;
    this.el.innerHTML=this._html;
}


test.html


<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>vue双向绑定原理</title>
</head>
<body>
    <div id="app"></div>
</body>
<script src="test1.js"></script>
<script>
    var vm = new vue();
    setTimeout(function(){
        console.log('changes');
        console.log(vm.$data);
        vm.$data.a=222;
    },2000)
</script>
</html>


3、Proxy还可以做什么呢?


(1)、校验类型


function createValidator(target,validator){
    return new Proxy(target,{
        _validator:validator,
        set:function(target,key,value,proxy){
            if (target.hasOwnProperty(key)) {
                var validator1=this._validator[key];
                if(validator1(value)){
                    return Reflect.set(target,key,value,proxy)
                }
                else{
                    throw Error('type error')
                }
            }
        }
    })
}
var personvalidator={
    name(val){
        return typeof val==='string'
    },
    age(val){
        return typeof val==='number'&&val>18
    }
}
class Person {
    constructor(name,age) {
        this.name=name;
        this.age=age;
        return createValidator(this,personvalidator);
    }
}
var tss=new Person('maomin',22);


(2)、真正私有变量


var api = {
    _secret: 'xxxx',
    _otherSec: 'bbb',
    ver: 'v0.0.1'
};
api = new Proxy(api, {
    get: function (target, key) {
        // 以 _ 下划线开头的都认为是 私有的
        if (key.startsWith('_')) {
            console.log('私有变量不能被访问');
            return false;
        }
        return target[key];
    },
    set: function (target, key, value) {
        if (key.startsWith('_')) {
            console.log('私有变量不能被修改');
            return false;
        }
        target[key] = value;
    },
    has: function (target, key) {
        return key.startsWith('_') ? false : (key in target);
    }
});
api._secret; // 私有变量不能被访问
console.log(api.ver); // v0.0.1
api._otherSec = 3; // 私有变量不能被修改
console.log('_secret' in api); //false
console.log('ver' in api); //true



目录
打赏
0
0
0
1
1
分享
相关文章
|
7天前
|
vue实现任务周期cron表达式选择组件
vue实现任务周期cron表达式选择组件
51 4
Vue实现动态数据透视表(交叉表)
Vue实现动态数据透视表(交叉表)
147 13
vue2和vue3的响应式原理有何不同?
大家好,我是V哥。本文详细对比了Vue 2与Vue 3的响应式原理:Vue 2基于`Object.defineProperty()`,适合小型项目但存在性能瓶颈;Vue 3采用`Proxy`,大幅优化初始化、更新性能及内存占用,更高效稳定。此外,我建议前端开发者关注鸿蒙趋势,2025年将是国产化替代关键期,推荐《鸿蒙 HarmonyOS 开发之路》卷1助你入行。老项目用Vue 2?不妨升级到Vue 3,提升用户体验!关注V哥爱编程,全栈开发轻松上手。
极致的灵活度满足工程美学:用Vue Flow绘制一个完美流程图
本文介绍了使用 Vue Flow 绘制流程图的方法与技巧。Vue Flow 是一个灵活强大的工具,适合自定义复杂的流程图。文章从环境要求(Node.js v20+ 和 Vue 3.3+)、基础入门案例、自定义功能(节点与连线的定制、事件处理)到实际案例全面解析其用法。重点强调了 Vue Flow 的高度灵活性,虽然预定义内容较少,但提供了丰富的 API 支持深度定制。同时,文中还分享了关于句柄(handles)的使用方法,以及如何解决官网复杂案例无法运行的问题。最后通过对比 mermaid,总结 Vue Flow 更适合需要高度自定义和复杂需求的场景,并附带多个相关技术博客链接供进一步学习。
管理数据必备;侦听器watch用法详解,vue2与vue3中watch的变化与差异
一篇文章同时搞定Vue2和Vue3的侦听器,是不是很棒?不要忘了Vue3中多了一个可选项watchEffect噢。 博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
属性描述符初探——Vue实现数据劫持的基础
属性描述符还有很多内容可以挖掘,比如defineProperty与Proxy的区别,比如vue2与vue3实现数据劫持的方式有什么不同,实现效果有哪些差异等,这篇博文只是入门,以后有时间再深挖。 博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
4月前
|
vue使用iconfont图标
vue使用iconfont图标
211 1
Vue Router 核心原理
Vue Router 是 Vue.js 的官方路由管理器,用于实现单页面应用(SPA)的路由功能。其核心原理包括路由配置、监听浏览器事件和组件渲染等。通过定义路径与组件的映射关系,Vue Router 将用户访问的路径与对应的组件关联,支持哈希和历史模式监听 URL 变化,确保页面导航时正确渲染组件。
ry-vue-flowable-xg:震撼来袭!这款基于 Vue 和 Flowable 的企业级工程项目管理项目,你绝不能错过
基于 Vue 和 Flowable 的企业级工程项目管理平台,免费开源且高度定制化。它覆盖投标管理、进度控制、财务核算等全流程需求,提供流程设计、部署、监控和任务管理等功能,适用于企业办公、生产制造、金融服务等多个场景,助力企业提升效率与竞争力。
170 12