Vue数据响应使用步骤详解

这次给大家带来Vue数据响应使用步骤详解,使用Vue数据响应的注意事项有哪些,下面就是实战案例,一起来看一下。

前言

Vue的数据响应主要是依赖了Object.defineProperty(),那么整个过程是怎么样的呢?以我们自己的想法来走Vue的道路,其实也就是以Vue的原理为终点,我们来逆推一下实现过程。

本文代码皆为低配版本,很多地方都不严谨,比如 if(typeof obj === ‘object’)这是在判断obj是否为为一个对象,虽然obj也有可能是数组等其他类型的数据,但是本文为了简便,就直接这样写来表示判断对象,对于数组使用Array.isArray()。

改造数据

我们先来尝试写一个函数,用于改造对象:

为什么要先写这个函数呢? 因为改造数据是一个最基础也是最重要的步骤,之后所有的步骤都会依赖这一步。

// 代码 1.1function defineReactive (obj,key,val) { Object.defineProperty(obj,key,{  enumerable: true,  configurable: true,  get: function () {   return val;  },  set: function (newVal) {   //判断新值与旧值是否相等   //判断的后半段是为了验证新值与旧值都为NaN的情况 NaN不等于自身   if(newVal === val || (newVal !== newVal && value !== value)){    return ;   }   val = newVal;  } });}

登录后复制

例如const obj = {},然后再调用defineReactive(obj,’a’,2)方法,此时在函数内,val=2,然后每次获取obj.a的值的时候都是获取val的值,设置obj.a的时候也是设置val的值。(每次调用defineReactive都会产生一个闭包保存了val的值);

流程讨论

经过验证之后,发现这个函数确实可以使用的。然后我们来讨论一下响应的流程:

Vue数据响应使用步骤详解

输入数据

改造数据(defineReactive())

如果数据变动 => 触发事件

我们来看第三步,数据变动如何触发之后的事件呢?仔细思考一下,如果要改变数据,那么必须先set数据,那么我们直接set()里面添加方法就ok了呀。

然后还有一个重要问题:

Vue数据响应使用步骤详解

依赖收集

我们怎么知道数据改变之后要触发的是什么事件呢?在Vue中:

使用数据 => 视图; 使用了数据来渲染视图,那么在获取数据的时候收集依赖是最佳的时机,Vue在改造数据属性的时候生成一个Dep实例,用于收集依赖。

// 代码 1.2class Dep { constructor(){  //订阅的信息  this.subs = []; } addSub(sub){  this.subs.push(sub); } removeSub (sub) {  remove(this.subs, sub); } //此方法的作用等同于 this.subs.push(Watcher); depend(){  if (Dep.target) {   Dep.target.addDep(this);  } } //这个方法就是发布通知了 告诉你 有改变啦 notify(){  const subs = this.subs.slice()  for (let i = 0, l = subs.length; i < l; i++) {   subs[i].update();  } }}Dep.target = null;

登录后复制

代码1.2就是Dep的部分代码,暂时只需要知道2个方法的作用就可以了

depend() — 可以理解为收集依赖的事件,不考虑其他方面的话 功能等同于addSub()

notify() — 这个方法更为直观了,执行所有依赖的update()方法。就是之后的改变视图啊 等等。

本篇主要讨论数据响应的过程,不深入讨论 Watcher类,所以Dep中的方法知道作用就可以了。

然后就是改变代码1.1了

//代码 1.3function defineReactive (obj,key,val) { const dep = new Dep(); Object.defineProperty(obj,key,{  enumerable: true,  configurable: true,  get: function () {   if(Dep.target){    //收集依赖 等同于 dep.addSub(Dep.target)    dep.depend()   }   return val;  },  set: function (newVal) {   if(newVal === val || (newVal !== newVal && val !== val)){    return ;   }   val = newVal;   //发布改变   dep.notify();  } });}

登录后复制

这代码中有一个疑点,Dep.target是什么?为什么要有Dep.target才会收集依赖呢?

Dep是一个类,Dep.target是类的属性,并不是dep实例的属性。

Dep类在全局可用,所以Dep.target在全局能访问到,可以任意改变它的值。

get这个方法使用很平常,不可能每次使用获取数据值的时候都去调用dep.depend()。

dep.depend()实际上就是dep.addSub(Dep.target)。

那么最好方法就是,在使用之前把Dep.target设置成某个对象,在订阅完成之后设置Dep.target = null。

验证

是时候来验证一波代码的可用性了

//代码 1.4const obj = {};//这一句是不是感觉很熟悉 就相当于初始化vue的data ---- data:{obj:{}};//低配的不能再低配的watcher对象(源码中是一个类,我这用一个对象代替了)const watcher = { addDep:function (dep) {  dep.addSub(this); }, update:function(){  html(); }}//假装这个是渲染页面的function html () { document.querySelector('body').innerHTML = obj.html;}defineReactive(obj,'html','how are you');//定义响应式的数据Dep.target = watcher;html();//第一次渲染界面Dep.target = null;

登录后复制

此时浏览器上的界面是这样的

Vue数据响应使用步骤详解

然后在下打开了控制台开始调试,输入:

obj.html = 'I am fine thank you'

登录后复制

然后就发现,按下回车的那一瞬间,奇迹发生了,页面变成了

Vue数据响应使用步骤详解

结尾

Vue数据响应的设计模式和订阅发布模式有一点像,但是不同,每一个dep实例就是一个订阅中心,每一次发布都会把所有的订阅全部发布出去。

Vue的响应式原理其实还有很大一部分,本文主要讨论了Vue是如何让数据进行响应,但是实际上,一般的数据都是很多的,一个数据被多处使用,改变数据之后观察新值,如何观察、如何订阅、如何调度,都还有很大一部分没有讨论。主要的三个类Dep(收集依赖)、Observer(观察数据)、Watcher(订阅者,若数据有变化通知订阅者),都只提了一点点。

之前写有一篇Vue响应式—-数组变异方法,针对Vue中对数组的改造进行讨论。当然之后有更多其他的文章,整个数据响应流程还有很多内容,三个主要的类都还没有讨论完。

其实阅读源码不仅仅是为了知道源码是如何工作的,更重要的是学习作者的思路与方法,我写的文章都不长,希望自己能够每次专注一个点,能够真真实实领悟到这一个点的原理。当然也想控制阅读时间,免得大家看到一半就关闭了。

相信看了本文案例你已经掌握了方法,更多精彩请关注【创想鸟】其它相关文章!

推荐阅读:

怎样使用Vue实现树形视图数据

JS对DOM树实现遍历有哪些方法

以上就是Vue数据响应使用步骤详解的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2753044.html

(0)
上一篇 2025年3月8日 06:32:49
下一篇 2025年2月21日 17:21:33

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • JS内this指向使用实例详解

    这次给大家带来JS内this指向使用实例详解,JS内this指向使用的注意事项有哪些,下面就是实战案例,一起来看一下。 在具体的实际应用中,this 的指向无法在函数定义时确定,而是在函数执行的时候才确定的,根据执行时的环境大致可以分为以下…

    编程技术 2025年3月8日
    200
  • es6中class特性使用案例详解

    这次给大家带来es6中class特性使用案例详解,es6中class特性使用的注意事项有哪些,下面就是实战案例,一起来看一下。 javaScript 语言中,生成实例对象的传统方法是通过构造函数,与传统的面向对象语言(比如 C++ 和 Ja…

    编程技术 2025年3月8日
    200
  • JS加载方式使用汇总

    这次给大家带来JS加载方式使用汇总,JS加载方式使用的注意事项有哪些,下面就是实战案例,一起来看一下。 一:同步加载 我们平时使用的最多的一种方式。 同步模式,又称阻塞模式,会阻止浏览器的后续处理,停止后续的解析,只有当当前加载完成,才能进…

    编程技术 2025年3月8日
    200
  • 如何使用redux-saga,使用redux-saga又有哪些方法和技巧?

    这篇文章主要介绍了redux-saga 初识和使,现在分享给大家,也给大家做个参考。 redux-saga 是一个管理 Redux 应用异步操作的中间件,功能类似redux-thunk + async/await, 它通过创建 Sagas …

    编程技术 2025年3月8日
    200
  • JS反射与依赖注入使用案例分析

    这次给大家带来JS反射与依赖注入使用案例分析,JS反射与依赖注入使用的注意事项有哪些,下面就是实战案例,一起来看一下。 对于javascript中的反射的理解,一直都是认为,利用数组对回调函数进行保存,之后在适当的时刻利用call或是app…

    编程技术 2025年3月8日
    200
  • 在vue项目中,使用axios跨域处理

    下面我就为大家分享一篇在vue项目中,使用axios跨域处理,具有很好的参考价值,希望对大家有所帮助。 跨域,一个很是让人尴尬的问题,有些人可以在后台中设置请求头,但是很多前端并不具备后台的知识,并无法自己独立的搭建一个服务器,所以就变成了…

    2025年3月8日
    200
  • 关于在使用vue.js标签属性中插入变量参数的方法(详细教程)

    这篇文章主要介绍了vue.js在标签属性中插入变量参数的方法,非常不错,具有参考借鉴价值,需要的朋友可以参考下 html的标签的属性,比如id、class、href需要动态传递参数,拼接字符串,查了一些资料,并没有找到合适的解决方法,琢磨了…

    2025年3月8日
    200
  • JS获取月具体天数步骤详解

    这次给大家带来JS获取月具体天数步骤详解,JS获取月具体天数的注意事项有哪些,下面就是实战案例,一起来看一下。 Javascript里面的new  Date(“xxxx/xx/xx”)这个日期的构造方法有一个妙处,当…

    编程技术 2025年3月8日
    200
  • node跨域使用express+http-proxy-middleware步骤详解

    这次给大家带来node跨域使用express+http-proxy-middleware步骤详解,node跨域使用express+http-proxy-middleware的注意事项有哪些,下面就是实战案例,一起来看一下。 为什么要使用no…

    编程技术 2025年3月8日
    200
  • 如何操作Vue内使用vee-validate

    这次给大家带来如何操作Vue内使用vee-validate,操作Vue内使用vee-validate的注意事项有哪些,下面就是实战案例,一起来看一下。 Vue项目遇到要表单验证了吧,对我来说表单验证是个很纠(dan)结(teng)的内容,各…

    2025年3月8日 编程技术
    200

发表回复

登录后才能评论