vue proxy 的优势与使用场景实现


Posted in Javascript onJune 15, 2020

1.前言

随着 vue3.x 的消息越来越多, proxy 的讨论也。相对于 Object.definePropertyproxy 有什么区别,有什么优势,以及可以应用在什么地方。该文章就简单的介绍下

2.Object.defineProperty

proxy 之前,先回顾下 Object.defineProperty 。大家都知道, vue2.x 以及之前的版本是使用 Object.defineProperty 实现数据的双向绑定的,至于是怎样绑定的呢?下面简单实现一下

function observer(obj) {
  if (typeof obj ==='object') {
    for (let key in obj) {
      defineReactive(obj, key, obj[key])
    }
  }
}

function defineReactive(obj, key, value) {
  //针对value是对象,递归检测
  observer(value)
  //劫持对象的key
  Object.defineProperty(obj, key, {
    get() {
      console.log('获取:'+key)
      return value
    },
    set(val) {
      //针对所设置的val是对象
      observer(val)
      console.log(key+"-数据改变了")
      value = val
    }
  })
}

let obj={
  name:'守候',
  flag:{
    book:{
      name:'js',
      page:325
    },
    interest:['火锅','旅游'],
  }
}

observer(obj)

在浏览器的 console 执行一下,似乎能正常运行

vue proxy 的优势与使用场景实现

但是实际上, Object.defineProperty 问题有以下几个

问题1.删除或者增加对象属性无法监听到

比如增加一个属性 gender ,由于在执行 observer(obj) 的时候,没有这个属性,所以这个无法监听到。删除的属性也是无法监听到

增加属性的时候, vue 需要使用 $set 进行操作, $set 的内部也是使用 Object.defineProperty 进行操作

vue proxy 的优势与使用场景实现

问题2.数组的变化无法监听到

vue proxy 的优势与使用场景实现

由上图得知,虽然数组属性实际上是修改成功了,但是不能被监听到

问题3. 由于是使用递归遍历对象,使用 Object.defineProperty 劫持对象的属性,如果遍历的对象层级比较深,花的时间比较久,甚至有性能的问题

3.proxy

对于 proxy ,在 mdn 上的描述是: 对象用于定义基本操作的自定义行为(如属性查找、赋值、枚举、函数调用等)

简单来说就是,可以在对目标对象设置一层拦截。无论对目标对象进行什么操作,都要经过这层拦截

听上去似乎, proxyObject.defineProperty 要好用,并且简单很多,实际上就是如此。下面用 proxy 对上面的代码进行改写试下

function observerProxy(obj){
  let handler = {
   get (target, key, receiver) {
    console.log('获取:'+key)
    // 如果是对象,就递归添加 proxy 拦截
    if (typeof target[key] === 'object' && target[key] !== null) {
     return new Proxy(target[key], handler)
    }
    return Reflect.get(target, key, receiver)
   },
   set (target, key, value, receiver) {
    console.log(key+"-数据改变了")
    return Reflect.set(target, key, value, receiver)
   }
  }
  return new Proxy(obj, handler)
}


let obj={
  name:'守候',
  flag:{
    book:{
      name:'js',
      page:325
    },
    interest:['火锅','旅游'],
  }
}

let objTest=observerProxy(obj)

也是一样的效果

vue proxy 的优势与使用场景实现

而且,能做到 Object.defineProperty 做不到的事情,比如增加一个属性 gender ,能够监听到

vue proxy 的优势与使用场景实现

操作数组,也能监听到

vue proxy 的优势与使用场景实现

最后敲一下黑板,简单总结一下两者的区别

1. Object.defineProperty 拦截的是对象的属性,会改变原对象。 proxy 是拦截整个对象,通过 new 生成一个新对象,不会改变原对象。

2. proxy 的拦截方式,除了上面的 get 和 set ,还有 11 种。选择的方式很多 Proxy ,也可以监听一些 Object.defineProperty 监听不到的操作,比如监听数组,监听对象属性的新增,删除等。

4.proxy 使用场景

关于 proxy 的使用场景,受限于篇幅,这里就简单列举几个,更多的可以移步我的 github 笔记或者 mdn 。

看到这里,两者的区别,和 proxy 的优势已经知道个大概了。但是在开发上,有哪些场景可以使用到 proxy 呢,下面列举个可能会遇上的情况

4-1.负索引数组

在使用 splice(-1)slice(-1) 等 API 的时候,当输入负数的时候,会定位到数组的最后一项,但是在普通数组上,并不能使用负数。 [1,2,3][-1] 这个代码并不能输出 3 。要让上面的代码输出 3 , 也可以使用 proxy 实现。

let ecArrayProxy = {
 get (target, key, receiver) {
  let _index=key<0?target.length+Number(key):key
  return Reflect.get(target, _index, receiver)
 }
}
let arr=new Proxy([1,2,3],ecArrayProxy)

vue proxy 的优势与使用场景实现 

4-2.表单校验

在对表单的值进行改动的时候,可以在 set 里面进行拦截,判断值是否合法

let ecValidate = {
 set (target, key, value, receiver) {
  if (key === 'age') {
   //如果值小于0,或者不是正整数
   if (value<0||!Number.isInteger(value)) {
    throw new TypeError('请输入正确的年龄');
   }
  }
  return Reflect.set(target, key, value, receiver)
 }
}

let obj=new Proxy({age:18},ecValidate)
obj.age=16
obj.age='少年'

vue proxy 的优势与使用场景实现 

4-3.增加附加属性

比如有一个需求,保证用户输入正确身份证号码之后,把出生年月,籍贯,性别都添加进用户信息里面

众所周知,身份证号码第一和第二位代表所在省(自治区,直辖市,特别行政区),第三和第四位代表所在市(地级市、自治州、盟及国家直辖市所属市辖区和县的汇总码)。第七至第十四位是出生年月日。低17位代表性别,男单女双。

const PROVINCE_NUMBER={
  44:'广东省',
  46:'海南省'
}
const CITY_NUMBER={
  4401:'广州市',
  4601:'海口市'
}

let ecCardNumber = {
 set (target, key, value, receiver) {
  if(key === 'cardNumber'){
    Reflect.set(target, 'hometown', PROVINCE_NUMBER[value.substr(0,2)]+CITY_NUMBER[value.substr(0,4)], receiver)
    Reflect.set(target, 'date', value.substr(6,8), receiver)
    Reflect.set(target, 'gender', value.substr(-2,1)%2===1?'男':'女', receiver)
  }
  return Reflect.set(target, key, value, receiver)
 }
}
let obj=new Proxy({cardNumber:''},ecCardNumber)

vue proxy 的优势与使用场景实现 

4-4.数据格式化

比如有一个需求,需要传时间戳给到后端,但是前端拿到的是一个时间字符串,这个也可以用 proxy 进行拦截,当得到时间字符串之后,可以自动加上时间戳。

let ecDate = {
 set (target, key, value, receiver) {
  if(key === 'date'){
    Reflect.set(target, 'timeStamp', +new Date(value), receiver)
  }
  return Reflect.set(target, key, value, receiver)
 }
}
let obj=new Proxy({date:''},ecDate)

vue proxy 的优势与使用场景实现 

参考链接

Proxy

面试官: 实现双向绑定Proxy比defineproperty优劣如何?

简单通俗的理解Vue3.0中的Proxy

小结

proxyObject.defineproperty 的一些区别,以及 proxy 的优势,使用场景,暂时就介绍到这里了。这篇文章介绍的不算深入,理解圈起来不难。对于 proxy ,笔者也打算继续深入学习,如果往后有收获,也会第一时间分享。如果文章有什么错误,或者有什么建议,欢迎评论区留言。

到此这篇关于vue proxy 的优势与使用场景实现的文章就介绍到这了,更多相关vue proxy 使用场景内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Javascript 相关文章推荐
基于jQuery的倒计时插件代码
May 07 Javascript
js判断上传文件类型判断FileUpload文件类型代码
May 20 Javascript
JQuery仿小米手机抢购页面倒计时效果
Dec 16 Javascript
javascript查询字符串参数的方法
Jan 28 Javascript
JS实现让网页背景图片斜向移动的方法
Feb 25 Javascript
javascript实现鼠标拖动改变层大小的方法
Apr 30 Javascript
初步使用Node连接Mysql数据库
Mar 03 Javascript
微信小程序 网络API Websocket详解
Nov 09 Javascript
使用axios实现上传图片进度条功能
Dec 21 Javascript
element-ui上传一张图片后隐藏上传按钮功能
May 22 Javascript
JavaScript ES6 Class类实现原理详解
May 08 Javascript
详解node.js创建一个web服务器(Server)的详细步骤
Jan 15 Javascript
原生JS利用transform实现banner的无限滚动示例代码
Jun 15 #Javascript
koa中间件核心(koa-compose)源码解读分析
Jun 15 #Javascript
为react组件库添加typescript类型提示的方法
Jun 15 #Javascript
JavaScript中的全局属性与方法深入解析
Jun 14 #Javascript
Vue使用Three.js加载glTF模型的方法详解
Jun 14 #Javascript
浅谈Vue 自动化部署打包上线
Jun 14 #Javascript
JS定时器如何实现提交成功提示功能
Jun 12 #Javascript
You might like
PHP新手上路(十)
2006/10/09 PHP
php网站来路获取代码(针对搜索引擎)
2010/06/08 PHP
一个非常实用的php文件上传类
2017/07/04 PHP
jQuery 在光标定位的地方插入文字的插件
2012/05/10 Javascript
Jquery 的outerHeight方法使用介绍
2013/09/11 Javascript
jQuery 文本框得失焦点的简单实例
2014/02/19 Javascript
JavaScript+CSS实现仿Mootools竖排弹性动画菜单效果
2015/10/14 Javascript
easyui 中的datagrid跨页勾选问题的实现方法
2017/01/18 Javascript
angular学习之ngRoute路由机制
2017/04/12 Javascript
基于Vue.js实现tab滑块效果
2017/07/23 Javascript
HTML5开发Kinect体感游戏的实例应用
2017/09/18 Javascript
jquery 实现拖动文件上传加载进度条功能
2018/03/18 jQuery
Vue 获取数组键名的方法
2018/06/21 Javascript
详解vue中的父子传值双向绑定及数据更新问题
2019/06/13 Javascript
Python转码问题的解决方法
2008/10/07 Python
python两种遍历字典(dict)的方法比较
2014/05/29 Python
在Python中执行系统命令的方法示例详解
2017/09/14 Python
Python中用psycopg2模块操作PostgreSQL方法
2017/11/28 Python
PyQt5每天必学之单行文本框
2018/04/19 Python
解决pycharm中导入自己写的.py函数出错问题
2020/02/12 Python
Python实现链表反转的方法分析【迭代法与递归法】
2020/02/22 Python
python3 中使用urllib问题以及urllib详解
2020/08/03 Python
纯CSS3实现鼠标悬停提示气泡效果
2014/02/28 HTML / CSS
css3 transform 3d 使用css3创建动态3d立方体(html5实践)
2013/01/06 HTML / CSS
HTML5通过navigator.mediaDevices.getUserMedia调用手机摄像头问题
2020/04/27 HTML / CSS
伦敦一卡通:The London Pass
2018/11/30 全球购物
美国婴儿和儿童服装购物网站:PatPat
2020/10/01 全球购物
前台领班岗位职责
2013/12/04 职场文书
资产经营总监岗位职责
2013/12/04 职场文书
电脑售后服务承诺书
2014/03/27 职场文书
出国留学担保书
2014/05/20 职场文书
中药学专业求职信
2014/05/31 职场文书
学校党员对照检查材料
2014/08/28 职场文书
基于Nginx实现限制某IP短时间访问次数
2021/03/31 Servers
JVM钩子函数的使用场景详解
2021/08/23 Java/Android
解决linux下redis数据库overcommit_memory问题
2022/02/24 Redis