本篇文章主要想简单聊聊vue如何实现数据修改,页面联动的底层原理
当然,篇幅有限,只是自己一些浅显的认知而已,我会从一下几个方面去聊,希望对你有所帮助。
  • 几个基础知识点
  • 数据代理
  • 数据劫持
  • 完整demo
 

一、几个基础知识点

1.普通函数和箭头函数的区别

我们知道,每个函数执行都会形成一条作用域链[[scopes]],函数内的所有变量其实都是在这条链上找的。

如上图所示,a函数定义在全局,其作用域链,只有GO对象,当其执行的时候会临时产生一个aAO对象,所以b函数的作用域链就是 aAO -> GO

函数每次执行都会产生一个新的AO对象挂在作用域链头部,函数被解释执行的时候,其内部标识符的检索都是在作用域链上检索的。

根据以上理论,我们来执行b函数。

控制台输出如下:

为什么b函数中看到的this是window呢?是因为其顺着作用域链找,bBO -> aAO -> GO,只有GO上有this,就是window。

当然对于普通函数,我们可以改变其this指向:

控制台输出如下:

于是我们得到一个结论:

函数执行生成的临时AO对象中,包含了arguments隐式变量来保存实参列表。

函数执行看到的this变量,可以修改,通过对象调用,call,apply来修改。

但是,但是,但是。。。。

箭头函数,它就不是这样的。。。

控制台输出:

箭头函数,没有arguments隐式变量了。而且,this它居然修改不了。。

那么说白了,this只能在其作用域链上找了,生成的临时AO对象上没有this,没有this。

所以:
  是否有arguments隐式变量 是否能改变this指向
普通函数
箭头函数
 
 
 
 
 
 

2.闭包

 
其实,理解了作用域链就理解了闭包。

由于plus,minus,showCount三个函数的作用域链中有aaa的AO对象,所以当他们被返回后,形成了闭包。

所以三个函数都能看见count变量。
 
 
 

3.defineProperty函数的使用

该函数可以说是Vue2中底层实现的基础。
我们一般定义对象都像如下这样:

但是其实除了这样给对象加属性外,我们也可以通过defineProperty来给对象加属性。

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>demo01-defineProperty的使用</title>
</head>
<body>
<script type="application/javascript">
// defineProperty()
let obj = {
name: 'zhangsan',
age: 33,
showInfo() {
console.log(this.name + "--" + this.age)
}
}
obj.showInfo(); Object.defineProperty(obj, 'ccc', {
value: 10,
enumerable: true, // 是否能枚举
configurable: true, // 是否能删除
writable: true // 是否能写入
})
// 枚举
var keys = Object.keys(obj);
console.log(keys);
// 写入
obj.ccc = 100;
console.log(obj);
// 删除
delete obj.ccc;
console.log(obj); </script>
</body>
</html>

上述代码控制台输入如下:

其实这样的话,定义属性和我们直接写属性没什么太大区别,关键是下面这样的写法:

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>demo01-defineProperty的使用</title>
</head>
<body>
<script type="application/javascript">
// defineProperty()
let obj = {
name: 'zhangsan',
age: 33,
showInfo() {
console.log(this.name + "--" + this.age)
}
} let ccc = 10;
Object.defineProperty(obj, 'ccc', {
// value: 10,
enumerable: true, // 是否能枚举
configurable: true, // 是否能删除
// writable: true, // 是否能写入
get: function proxyGet() {
return ccc;
},
set: function proxySet(value) {
ccc = value;
}
})
console.log(obj.ccc);
obj.ccc = 100;
console.log(obj.ccc); console.log(obj); </script>
</body>
</html>

注意,如果我们要定义属性的get/set,那么就不能定义value和writable了,否则会报错。

此时我们对属性ccc的写入和读取将走get/set方法了。

控制台输出如下:

这个ccc属性的三个点,是不是特别想我们使用vue的时候点开的组件对象里面的一些属性。

在这里我插一句,我点开set/get给大家看看,其实能看见[[scopes]]作用域链了。

如下图:

当然,这里我们看见了,set/get函数定义的时候的作用域链[[scopes]],其实是SO -> GO,这个SO其实就是外层包裹的script标签。

可以理解成,script标签执行流程就像一个函数执行一样,也会产生作用域对象挂在[[scopes]]上。

其实讲到这里,这个set/get是数据代理和劫持的关键。
 
 

二、数据代理

 
再讲数据代理之前,我们可以使用vue来写个demo,目的看看我们每次配置组件的时候,data对象去哪里了?
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>demo01-vue简单使用</title>
<!--引入vue-->
<script src="https://cdn.jsdelivr.net/npm/vue@2.7.14/dist/vue.js"></script> </head>
<body> <div id="app">
<h3>姓名:{{name}}</h3>
<h3>年龄:{{age}}</h3>
<button @click="agePlusOne">年龄+1</button>
</div> <script type="application/javascript"> let vm = new Vue({
el: '#app',
data(){
return {
name: '张三',
age: 33
}
},
methods: {
agePlusOne(){
this.age ++;
console.log(this)
}
}
}); </script>
</body>
</html>

点击按钮,我把Vue对象打印出来:

可以清晰的看到,我们配置的data对象中的属性都被定义在了Vue组件对象中。

起码,这里看到了,vue做了数据代理,我们在组件对象中对data中同名属性的set和get都走了其对应的代理方法。

到这里我们可以这样理解,options.data对象传入之后,vue生成了一个_data对象挂在了实例身上,而且vm._data === options.data。
然后,vue通过defineProperty方法在实例身上定义了data中定义的属性,并set/get都指向了_data中的对应属性。
 

三、数据劫持

我理解的数据劫持,就是属性在设置或者获取的时候,做点什么?
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>demo01-数据劫持</title>
</head>
<body>
<div id="app">
</div>
<script type="application/javascript">
function setAppInnerText (value) {
document.querySelector("#app").innerText = value;
} let obj = {
name: '张三',
age: 100,
showInfo() {
return this.name + "---" + this.age;
}
}; setAppInnerText(obj.showInfo())
// 数据劫持
Object.keys(obj).forEach(key => {
let value = obj[key];
Object.defineProperty(obj, key, {
enumerable: true,
configurable: true,
set(newValue) {
if(newValue === value) {
return ;
} else {
value = newValue;
setAppInnerText(obj.showInfo());
}
},
get() {
return value;
}
})
})
obj.age = 1000;
</script>
</body>
</html>

上面的代码,就是数据劫持,每次属性设置的时候,都触发了setAppInnerText函数的调用。

上述代码执行完之后,只要我们对obj对象的属性进行修改,都会触发页面的变化。

 
 

四、完整demo

当然,任何一个框架的代码都是很复杂的,因为要考虑很多东西。
这里我只是基于自己的理解,写一个简单的demo,目的只是为了帮助大家理解vue,理解它如何做到数据修改,页面变化的。
<!DOCTYPE html>
<html lang="en"> <head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>data_observer</title>
</head> <body>
<div id="root">
a = {{a}}
<br>
b = {{b}}
</div>
<script> function Vue(config) {
this._data = config.data;
// 数据代理 方便程序员操作
for (let key in config.data) {
Object.defineProperty(this, key, {
enumerable: true,
get: function proxyGet() {
return this._data[key];
},
set: function proxySet(value) {
this._data[key] = value;
}
})
}
this.mounted = false;
if (config.el) {
this.$mount(config.el);
}
} Vue.prototype.$mount = function (id) {
if (!this.mounted) {
this.originInnerHtml = document.getElementById(id).innerHTML;
// 编译模板生成render
let _self = this;
function render() {
let innerHtml = _self.originInnerHtml;
for (let key in _self._data) {
innerHtml = innerHtml.replaceAll('{{' + key + '}}', _self._data[key]);
}
document.getElementById(id).innerHTML = innerHtml;
} // 数据劫持
for (let key in this._data) {
let value = this._data[key];
Object.defineProperty(this._data, key, {
enumerable: true,
configurable: true,
get: function getObserver() {
return value;
},
set: function setObserver(newValue) {
if (value !== newValue) {
value = newValue;
render();
}
}
})
}
// 执行render
render();
this.mounted = true;
}
}
let config = {
el: 'root',
data: {
a: '牛逼的消息',
b: '学习vue2底层实现'
}
};
let vm = new Vue(config);
</script>
</body> </html>

控制台打印如下:

我相信到这里,你应该能理解vue2大体怎么实现响应式的了。
无非就是2点:
1.使用数据代理给每个组件实例添加属性,方便我们编程的时候操作;
2.使用数据劫持,监听对象每个层级属性的变化,内部触发重新渲染。
当然,真正的源码包含了对对象的每个层级的属性的监测,我这里只是简单写个demo,目的是为了方便大家理解。
数据劫持,内部肯定是闭包,使用闭包封装了每个属性,本篇也提到了作用域链[[scopes]]还有我们在编程中经常困惑的箭头函数。
总之,希望对大家有帮助吧。
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

vue2原理初探-数据代理和数据劫持的更多相关文章

  1. 对数据劫持 OR 数据代理 的研究------------引用

    数据劫持,也叫数据代理. 所谓数据劫持,指的是在访问或者修改对象的某个属性时,通过一段代码拦截这个行为,进行额外的操作或者修改返回结果.比较典型的是 Object.defineProperty() 和 ...

  2. vue中的数据代理原理

    const vm = new Vue({ data:{ name:'boos' } }) // 注意 :使用构造函数构建vue实例时,传入的是一个option对象,它包含了data,computed等 ...

  3. es6中的Proxy和vue中的数据代理的异同

    1:概述 1-1:Proxy 用于修改某些操作的默认行为,Proxy可以说在对对象进行各种访问或者操作的时候在外层进行一层拦截,在操作之前都需要经过这种拦截.proxy返回的是一个新对象,可以通过操作 ...

  4. 8_vue是如何进行数据代理的

    在了解了关于js当中的Object.defineProperty()这个方法后,我们继续对vue当中的数据代理做一个基于现在的解析 建议观看之前先了解下js当中的Obejct.defineProper ...

  5. Vue源码中的数据代理

    直接开讲: ​ 由于这个Vue底层封装的函数太多了,我这里只讲思路不说具体的执行了什么函数. ​ const vm=new Vue({这里写一个data,可以是对象也可以是函数}) 在写这段代码的时候 ...

  6. MVVM模型 && 数据代理

    MVVM模型 观察发现 data中所有属性,最后都出现在vm身上 vm身上所有属性及Vue原型身上所有属性,在Vue模板中都可以直接使用 Vue中的数据代理 通过vm对象来代理data对象中属性的操作 ...

  7. WCF初探-16:WCF数据协定之基础知识

    数据协定概念 “数据协定”是在服务与客户端之间达成的正式协议,用于以抽象方式描述要交换的数据. 也就是说,为了进行通信,客户端和服务不必共享相同的类型,而只需共享相同的数据协定. 数据协定为每一个做数 ...

  8. WCF初探-17:WCF数据协定之等效性

    数据协定等效性特点概述 对于客户端要将某种类型的数据成功发送到服务,或者服务要将数据成功发送到客户端的情况,接收端上并不一定必须存在此发送数据类型. 唯一的要求是两种类型的数据协定应该等效. 要使数据 ...

  9. WCF初探-18:WCF数据协定之KnownType

    KnownTypeAttribute 类概述 在数据到达接收终结点时,WCF 运行库尝试将数据反序列化为公共语言运行库 (CLR) 类型的实例.通过首先检查传入消息选择为反序列化而实例化的类型,以确定 ...

  10. ElasticSearch 学习记录之 分布式文档存储往ES中存数据和取数据的原理

    分布式文档存储 ES分布式特性 屏蔽了分布式系统的复杂性 集群内的原理 垂直扩容和水平扩容 真正的扩容能力是来自于水平扩容–为集群添加更多的节点,并且将负载压力和稳定性分散到这些节点中 ES集群特点 ...

随机推荐

  1. hosts文件妙用,提升网站访问速度!

    一.背景 在讲解hosts文件之前,我们先了解下IP地址与域名的关系. 1.IP地址与域名的关系 IP(Internet Protocol)是一种规定互联网中数据传输的协议,每台连接到互联网中的计算机 ...

  2. Spring Boot实现高质量的CRUD-3

    (续前文) 7.Service接口类 ​ ​ Service类提供业务的实现逻辑,其调用Dao类的方法进行数据存取,并为Controller类提供方法.类似于Dao的接口类,服务层使用接口类,便于代码 ...

  3. SpringBoot集成支付宝 - 少走弯路就看这篇

    最近在做一个网站,后端采用了SpringBoot,需要集成支付宝进行线上支付,在这个过程中研究了大量支付宝的集成资料,也走了一些弯路,现在总结出来,相信你读完也能轻松集成支付宝支付. 在开始集成支付宝 ...

  4. 现代C++学习指南-具体类

    类作为C++中重要的概念之一,有着众多的特性,也是最迷人的部分! 类是一个加工厂,开发者使用C++提供的各种材料组装这个工厂,使得它可以生产出符合自己要求的数据,通过对工厂的改造,可以精细控制对象从出 ...

  5. React后台管理系统11 配置项目初始化展开代码

    在上一文中,我们已经配置好了,刷新默认打开选中的样式,但是如果是在/page3/1,这种的,并没有选中到/page3里面的/page3/1,这个地方来,所以我们需要解决的就是这几个问题: 思路如下: ...

  6. 手写RPC框架之泛化调用

    一.背景 前段时间了解了泛化调用这个玩意儿,又想到自己之前写过一个RPC框架(参考<手写一个RPC框架>),于是便想小试牛刀. 二.泛化调用简介 什么是泛化调用 泛化调用就是在不依赖服务方 ...

  7. Redis基础(二)——列表操作、redis管道、Django中使用redis

    Redis列表操作 ''' lpush(name,values) rpush(name, values) 表示从右向左操作 lpushx(name,value) rpushx(name, value) ...

  8. PHP正则按照从大到小的SIGN签名算法

    <?php/** * 签名算法 * @param unknown $key_id S_KEY(商户KEY) * @param unknown $array 例子:$array = array(' ...

  9. 一文帮你搞定H5、小程序、Taro长列表曝光埋点

    对于很多前端同学来说,"埋点"常常是一个不愿面对却又无法逃避的话题.为什么这么说呢,相信很多前端同学都深有体会:首先埋点这个事基本是前端"独享"的,服务端基本不 ...

  10. 一张表实现类某音的视频"评论回复"功能

    ​前言 现如今,不管是哪种类型的应用,评论区都少不了.从工具类的到媒体信息流类的,评论留言都是最基本的互动环节.比如抖音短视频下,针对视频每个用户都可以发表自己的观点:而针对用户的评论,其他的用户又可 ...