vue --- 2.0数据的响应式的一种实现

初识:

  • 实际上是通过Object.defineProperty()方法来实现的
  • talk is cheap, show your code
let obj = {};
Object.defineProperty(obj, 'name', {get(){return document.querySelector('#name').innerHTML;},set(newVal){document.querySelector('#name').innerHTML = val;}
})// 注1: 以上代码,将dom中id为name的值与obj.name属性绑定.
// 注2: 当在js文件中调用obj.name时,会触发obj的get函数(返回html内容)
// 注3: 当调用obj.name = xxx 时,会调用 obj 的set方法 

封装:

  • 知道了基本原理后,下一步是要将其封装起来!
// 首先创建一个Lz类.
class Lz{constructor(options){// 将传入的数据保存this.$options = options;// 数据响应化this.$data = options.data;this.observe(this.$data);}// 说明:// 将数据保存到this.$data中// observe函数用于给每一个数据添加set和get方法observe(value){if(!value || typeof value !== 'object'){return;}// 遍历该对象Object.keys(value).forEach(key =>{this.defineReactive(value, key, value[key])})}// 给数据添加响应式defineReactive(obj, key, val){Object.defineProperty(obj, key,{get(){return val},set(newVal){if(val === newVal) return;// 处理更新事件,此处使用console.log实现console.log(`数据更新辣: ${val} --- > ${newVal}`);val = newVal;}})}
}

引用Lz类

  • 在html中通过script标签导入Lz类
<body><div id="app"></div><script src="./lz.js"></script><script>const app = new Lz({el:'#app',data:{foo:'bar',hello:{world:'您好,世界'}}})</script>
</body>
  • 打开浏览器,在控制台输入app.$data.foo
  • 尝试改变app.$data.foo的值
  • 尝试改变app.$data.hello.world的值
    在这里插入图片描述

浅拷贝

  • 以上,对第一个属性foo的操作成功的触发了set函数.
  • 由于浅拷贝复制的是引用,我们对对象的修改无法触发set.
  • 简单的理解就是set只负责监听自己这一层的变化,下一层的变化.不予监听
  • 解决办法,使用递归.给每一层添加一个setter方法.代码如下
  • 只需在defineReactive里面对world对象设置set和get
  • 重写defineReactive方法
defineReactive(obj, key, val){// 解决浅拷贝问题this.observe(val);Object.defineProperty(obj, key, {get(){return val;},set(newVal){if(newVal === val){return;}console.log(`${key}属性更新: ${val} --- > ${newVal}`);val = newVal;}})
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/250625.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

心得开始之路

本人是大四实习狗&#xff0c;现在最大的问题是什么&#xff1f; 一&#xff1a;实力不够 二&#xff1a;人又懒 开始以为就做做运维&#xff0c;学学服务器就可以了&#xff0c;但是现在才发现&#xff0c;嗯&#xff0c;不会开发的运维什么都不算。 现在开始学习Python自动运…

结对编程作业——毕设导师智能匹配

结对编程作业——毕设导师智能匹配 031402317 李佳恺031402511 黄家俊 问题描述及要求 输入30个老师&#xff08;包含带学生数的要求的上限&#xff0c;单个数值&#xff0c;在[0,8]内&#xff09;&#xff0c;100个学生&#xff08;包含绩点信息&#xff09;&#xff0c;每…

内置函数二

内置函数: 1.lambda 匿名函数 lambda 参数:返回值 例    resultlambda x,y:xy sresult(x3,y4) print(s) 2.sorted 排序 sorted(iterable, keyfunc, reverseTurn/False) 例    lst [1, 8, 18, 19, 97, 12, 3] lst.sort() lst自带的排序功能  l2 sorted(lst) 排序…

vue --- 2.0响应式补充

补充: 数组的响应式 // 对数组的方法进行重写 // 1. 不能影响本来的方法 // 2. 调用的时候可以找到它 let odlArrayPrototype Array.prototype; let proto Object.create(odlArrayPrototype); // 继承 [push,shift,unshift].forEach(method >{proto[method] function(){…

OptaPlanner - 把example运行起来(运行并浅析Cloud balancing)

经过上面篇长篇大论的理论之后&#xff0c;在开始讲解Optaplanner相关基本概念及用法之前&#xff0c;我们先把他们提供的示例运行起来&#xff0c;好先让大家看看它是如何工作的。OptaPlanner的优点不仅仅是提供详细丰富的文档 &#xff0c;还为各种应用场景提供丰富的示例&am…

es6 --- 使用proxy对数据进行劫持

说明: 数据劫持,简单的说就是在对数据进行操作(增删改查)时,触发的函数下面想通过使用以下的形式来使用: let proxy reactive({ name:lz }); proxy.name; // 获取 proxy.name 栗子; // 设置 delete proxy.name; // 删除解决方案: proxy函数的2个参数第一个参数: 接收一…

Java8-如何构建一个Stream

Stream的创建方式有很多种&#xff0c;除了最常见的集合创建&#xff0c;还有其他几种方式。 List转Stream List继承自Collection接口&#xff0c;而Collection提供了stream()方法。 List<Integer> list Lists.newArrayList(1, 2, 3); Stream<Integer> stream li…

软件产品案例分析

软件产品案例分析 第一部分&#xff1a; 评测&#xff1a; 上手体验&#xff1a; 说实话&#xff0c;在老师布置这个作业之前我确实不知道有K米这个APP&#xff0c;我想这是很少去KTV的原因吧。。。不过在接到这个作业后&#xff0c;我就去百度了普及了一下这个app的相关知识。…

java/android 做题中整理的碎片小贴士(12)

1、edittext中设置最长字数&#xff0c;可在xml中加入android:maxLength"10"&#xff0c;可在java代码中加入editText.setFilters(new InputFilter[]{new InputFilter.LengthFilter(20)}); 2、edittext监听软键盘delete按键&#xff1a; Et.setOnKeyListener(new Vi…

POJ2777(线段树裸题)

题目&#xff1a;http://poj.org/problem?id2777 别忘了各地的return&#xff1b; 有可能输入的L<R&#xff0c;手动swap&#xff1b; 似乎是多组输入&#xff1f; pushup和pushdown的位置。 &#xff08;原来pushup只有一行&#xff09; 要开四倍数组。是这种写法的原因吧…

vue --- 2.0 编译的实现

初识 假设html中有如下dom: <div id"app"><!-- 插值绑定 --><p>{{name}}</p><!-- 指令解析 --><p l-text"name"></p><p>{{age}}</p><p>{{doubleAge}}</p><!-- 双向绑定实现 -->…

个人作业收官——软件工程实践总结

一、回望与展望 1.1 对比现在和开学初博客开篇的课程目标和期待 当初的目标&#xff1a; 提升团队合作的能力能够学习到开发的一系列流程&#xff0c;以及如何写高质量的代码加强自己的编码能力&#xff0c;以及编码习惯熟悉不同平台的开发过程 如今&#xff1a; 基本的目标都…

sklearn中SVM调参说明

写在前面 之前只停留在理论上&#xff0c;没有实际沉下心去调参&#xff0c;实际去做了后&#xff0c;发现调参是个大工程&#xff08;玄学&#xff09;。于是这篇来总结一下sklearn中svm的参数说明以及调参经验。方便以后查询和回忆。 常用核函数 1.linear核函数: K(xi,xj)xTi…

TZOJ 3030 Courses(二分图匹配)

描述 Consider a group of N students and P courses. Each student visits zero, one or more than one courses. Your task is to determine whether it is possible to form a committee of exactly P students that satisfies simultaneously the conditions: every stude…

vue --- configureWebpack模拟后台数据

初识 使用vue/cli搭建的项目可以在vue.config.js中,模拟一个后台(express写法)vue.config.js configureWebpack: {devServer: {// 模拟后台服务器 express写法before(app) {app.get(/api/login, function(req, res) {const { username, passwd } req.query;console.log(user…

TCP和UDP的优缺点及区别

转自&#xff1a;http://www.cnblogs.com/xiaomayizoe/p/5258754.html TCP的优点&#xff1a; 可靠&#xff0c;稳定 TCP的可靠体现在TCP在传递数据之前&#xff0c;会有三次握手来建立连接&#xff0c;而且在数据传递时&#xff0c;有确认、窗口、重传、拥塞控制机制&#xff…

e.getMessage 为空NULL

e.getMessage 为空NULL 在日常代码中免不了要try catch 切忌用try catch 去try 整个方法。 在对象操作之前尽量写上if 空判断。 反例&#xff1a; public void send(){ try{ 代码1&#xff1a;获取对象 代码2&#xff1a;操作代码1 代码3&#xff1a;操作代码2 代码4&#xff1…

Linux:客户端的实现

写了一个简单的服务器软件&#xff0c;但是没有写客户端。现在我将客户端实现了&#xff0c;其实昨天已经说了客户端的实现步骤了。 步骤&#xff1a; socket() 初始化 connet()链接 从标准输入读数据fgets() 传数据到服务器write() 读从服务器返回的数据read() 写数据到屏幕上…

vue --- http拦截,登录登出的逻辑设计

设计 在src目录下创建一个interceptor.js登录逻辑 设置拦截,在发起请求前,先判断用户是否登录(在本栗中,即是否能够在浏览器缓存中找到token). 登出逻辑 对服务端传过来的数据进行拦截,判断其状态码是否为401(未登录或token过期)清空浏览器缓存中的token重定向到登入页面 inte…