一文大白话讲清楚ES6代理Proxy和反射Reflect

文章目录

  • 一文大白话讲清楚ES6代理Proxy和反射Reflect
  • 1. 你当过老板么
  • 2.代理Proxy
    • 2.1 get(target,propKey,receiver)//获取对象的属性
    • 2.2 set(target,propKey,newValue,receiver)//设置属性的值
    • 2.3 has(target,propKey)//代理查询属性操作,propKey in obj的操作
    • 2.4 deleteProperty(target,propKey),代理属性删除操作,delete obj.属性
    • 2.5 ownKeys(target)拦截属性查询操作,Object.keys(),Object.getOwnPropertyNames(),Object.getOwnPropertySymbols()等
    • 2.6 getOwnPropertyDescriptor 代理Object。getOwnPropertyDescriptor(obj,propKey)
    • 2.7 defineProperty(target,propKey,PropDesc),代理Object.defineProperty
    • 2.8 preventExtensions(target),代理:拦截 Object.preventExtensions(proxy),禁止对象可扩展
    • 2.9 isExtensible(target),代理Object.isExtensible
    • 2.10 getPrototypeOf(target),代理Object.getPrototypeOf(target)
    • 2.11 apply(target,object,args),代理Proxy作为函数调用的操作
    • 2.11 construct(target,args),代理new 操作
  • 3. 反射Reflect

一文大白话讲清楚ES6代理Proxy和反射Reflect

1. 你当过老板么

  • 老板很忙,很多事自己忙不过来,怎么办,找一个秘书,给秘书授权,以他的名义去做事。
  • 那简单了,老板能做的我都能做,这就是代理
  • 我做完了,总得汇报给老板,让老板信息同步吧,好嘛,这就是反射
  • 回到js,我们有一个对象A,不直接操作对象A,而是找来B,让B成为A的代理,然后去操作B。B操作完了再去同步到A。这就是代理和反射

2.代理Proxy

  • 用于定于对象基本操作的自定义行为
  • 说白了,就是修改程序的默认行为,想好比在编程语言层面上做修改,属于元编程
  • 再简单点理解,就是把操作对象的底层方法进行覆写
  • 啥意思,就是假设有一个对象,obj={name:‘tom’},正常情况下我们obj.name就能获取到tom。但是我现在不想获取到tom,我让obj.name干点别的,行不行,行,比如让obj.name返回shift
  • 上代码
let obj={name:'tom'
}
// 用proxy代理obj
let proxy=new Proxy(obj,{//proxy.name的返回值get(target,name,reciever){return 'shift'}
})
console.log(obj.name)//tom
console.log(proxy.name)//shift
  • 明白了吧,proxy的任务就是代理obj,
  • 那都能代理obj的哪些方法呢

2.1 get(target,propKey,receiver)//获取对象的属性

  • 上面举了例子了

2.2 set(target,propKey,newValue,receiver)//设置属性的值

let obj={name:'tom'
}
let proxy=new Proxy(obj,{set(target,propKey,newValue,reciever){//如果设置proxy.name,就执行下边这句话,把obj.name改为'hahaha'target[propKey]='hahaha'}
})
proxy.name='new Tom'//这里设置name的新值为newTOm
console.log(proxy.name)//hahaha,但最终打印出来的是hahaha

2.3 has(target,propKey)//代理查询属性操作,propKey in obj的操作

let obj={name:'name'
}
let proxy=new Proxy(obj,{has(target,propKey){if(propKey==='age'){return true}}
})
console.log(age in obj)//false,obj里面没有age属性,所以返回false
console.log(age in proxy)//true,proxy里面也灭有age,但是我们拦截覆写了has方法,让他返回rue

2.4 deleteProperty(target,propKey),代理属性删除操作,delete obj.属性

let obj={name:'name',age:'age',
}
let proxy=new Proxy(obj,{deleteProperty(target,propKey){if(propKey==='name'){Reflect.deleteProperty(target,propKey)//如果删除name属性,就正常删除}else if(propKey==='age'){return false//如果删除age属性,则直接返回false,表示删除失败}}
})
console.log(proxy)//{ name: 'name', age: 'age' }
delete proxy.name
delete proxy.age
console.log(proxy)//{ age: 'age' },age属性并没有删除掉

2.5 ownKeys(target)拦截属性查询操作,Object.keys(),Object.getOwnPropertyNames(),Object.getOwnPropertySymbols()等

let obj={name:'tom',age:12
}
let proxy=new Proxy(obj,{ownKeys(target){return ['d','f','g']}
})
console.log(Object.getOwnPropertyNames(proxy))//[ 'd', 'f', 'g' ],正常应该打印['name','age']

2.6 getOwnPropertyDescriptor 代理Object。getOwnPropertyDescriptor(obj,propKey)

let obj={name:'tom',age:12
}
let proxy=new Proxy(obj,{getOwnPropertyDescriptor(target,propKey){return{ value: 'ProxyTom', writable: true, enumerable: true, configurable: true }}
})
console.log(Object.getOwnPropertyDescriptor(proxy,'name'))//正常应该返回{ value: 'tom', writable: true, enumerable: true, configurable: true }
//实际返回
// {
//     value: 'ProxyTom',
//         writable: true,
//     enumerable: true,
//     configurable: true
// }

2.7 defineProperty(target,propKey,PropDesc),代理Object.defineProperty

let obj={name:'tom',age:12
}
let proxy=new Proxy(obj,{defineProperty(target,propKey,propDesc){return Object.defineProperty(target,propKey,{value:'166cm',writable:true,configurable:true,enumerable:true,})}
})
Object.defineProperty(proxy,'height',{value:'163cm',writable:true,configurable:true,enumerable:true,
})
console.log(proxy)//{ name: 'tom', age: 12, height: '166cm' },正常情况下应该打印163cm,是因为对defineProperty进行了覆写

2.8 preventExtensions(target),代理:拦截 Object.preventExtensions(proxy),禁止对象可扩展

let obj={name:'tom'
}
let proxy=new Proxy(obj,{preventExtesions(target){return false}
})
Object.preventExtensions(proxy)//设置对象不可扩展,这里会报错,阻止禁止扩展的行为

2.9 isExtensible(target),代理Object.isExtensible

let obj={name:'tom'
}
let proxy=new Proxy(obj,{isExtensible(target){return Reflect.isExtensible(target)},})
Object.preventExtensions(proxy)//设置对象不可扩展,这里会报错,阻止禁止扩展的行为
console.log(Object.isExtensible(proxy))//false

2.10 getPrototypeOf(target),代理Object.getPrototypeOf(target)

let obj={name:'tom'
}
let proxy=new Proxy(obj,{getPrototypeOf(target){return {age:12}},})console.log(Object.getPrototypeOf(proxy))//{ age: 12 },正常情况下应该打印{}

2.11 apply(target,object,args),代理Proxy作为函数调用的操作

let fn=function (){return 1
}
console.log(fn())//1
let proxy=new Proxy(fn,{apply(target,object,args){return 4}
})
let g=function (){}
console.log(proxy.call(g))//4

2.11 construct(target,args),代理new 操作

let fn=function (){this.name='tom'
}
let proxy=new Proxy(fn,{construct(target,args){return {name:'new-tom'}}
})
let p=new proxy()
console.log(p)//{ name: 'new-tom' }//正常打印{ name: 'tom' }

3. 反射Reflect

  • 上面我们都是在代理对象proxy上操作,源对象obj并没有得到同步
  • Reflect的作用就是把对proxy的操作同步到obj上
  • 每个Proxy代理的方法,Reflect都有同名对应的方法取反射
  • 上代码
let obj={name:'tom'
}
let proxy=new Proxy(obj,{set(target,propKey,newVale,receiver){Reflect.set(target,propKey,newVale,receiver)//同步映射到obj}
})
proxy.age=12;
console.log(proxy)
console.log(obj)
// { name: 'tom', age: 12 }
// { name: 'tom', age: 12 }
  • 其他2.1——2.11原理都是一样的,不做一一演示,大家自行写一下

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

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

相关文章

VS2022引入sqlite数据库交互

法一:用官网编译好的动态库(推荐) 下载所需文件 sqlite官网地址:https://www.sqlite.org/howtocompile.html 下载以下的2个压缩包 第一个压缩包 sqlite-amalgamation-xxxx.zip,xxxx是版本号,保持一致即可,这里面有sqite3.h 第…

Java后端常用的4种请求方式(通俗易懂)

文章目录 前言通用接口类(ControllerDemo)通用实体类(UserEntity)通用响应类(HttpClientResult)成功截图(先启动项目,然后右键执行main方法) HttpClientHttpClient 的主要类代码案例导入依赖工具类(HttpClientUtil)测试类 HttpURLConnection简介调用步骤代码案例导入依赖工具类…

STM32烧写失败之Contents mismatch at: 0800005CH (Flash=FFH Required=29H) !

一)问题:用ULINK2给STM32F103C8T6下载程序,下载方式设置如下: 出现下面两个问题: 1)下载问题界面如下: 这个错误的信息大概可以理解为,在0x08000063地址上读取到flash存储为FF&am…

Spring Boot教程之五十二:CrudRepository 和 JpaRepository 之间的区别

Spring Boot – CrudRepository 和 JpaRepository 之间的区别 Spring Boot建立在 Spring 之上,包含 Spring 的所有功能。由于其快速的生产就绪环境,使开发人员能够直接专注于逻辑,而不必费力配置和设置,因此如今它正成为开发人员…

【SQL】Delete使用

目录 语法 需求 示例 分析 代码 语法 DELETE删除表中所需内容 删除表中满足特点条件的行:DELETE FROM 表名 WHERE 条件; 删除表中所有行:DELETE FROM 表名; WHERE子句 WHERE子句用于指定从表中选取记录的条件。允许筛选数据,只返回满足…

【express-generator】05-路由中间件和错误处理(第一阶段收尾)

一、前言 上篇文章我们介绍了express-generator的请求体解析,重点讲了常用的请求体数据格式(JSON/URL 编码的表单数据)以及一个FILE文件上传,同时搭配代码示范进行辅助理解。 二、本篇重点 我们继续第一阶段的知识,…

Vue3(elementPlus) el-table替换/隐藏行箭头,点击整行展开

element文档链接: https://element-plus.org/zh-CN/component/form.html 一、el-table表格行展开关闭箭头替换成加减号 注:Vue3在样式中修改箭头图标无效,可能我设置不对,欢迎各位来交流指导 转变思路:隐藏箭头&…

GitLab创建用户,设置访问SSH Key

继上一篇 Linux Red Hat 7.9 Server安装GitLab-CSDN博客 安装好gitlab,启用管理员root账号后,开始创建用户账户 1、创建用户账户 进入管理后台页面 点击 New User 输入用户名、邮箱等必填信息和登录密码 密码最小的8位,不然会不通过 拉到…

数据库环境安装(day1)

网址:MySQL 下载(环境准备): (2-5点击此处,然后选择合适的版本) 1.linux在线YUM仓库 下载/安装: wget https://repo.mysql.com//mysql84-community-release-el9-1.noarch.rpm rpm -i https://r…

Qt 5.14.2 学习记录 —— 칠 QWidget 常用控件(2)

文章目录 1、Window Frame2、windowTitle3、windowIcon4、qrc机制5、windowOpacity 1、Window Frame 在运行Qt程序后,除了用户做的界面,最上面还有一个框,这就是window frame框。对于界面的元素,它们的原点是Qt界面的左上角或win…

牛客网刷题 ——C语言初阶(6指针)——BC105 矩阵相等判定

1. 题目描述:BC105 矩阵相等判定 牛客网OJ题链接 描述: KiKi得到了两个n行m列的矩阵,他想知道两个矩阵是否相等,请你回答他。(当两个矩阵对应数组元素都相等时两个矩阵相等)。 示例1 输入: 2 2 1 2 3 4 1 2 3 4 输出…

HTML5实现好看的中秋节网页源码

HTML5实现好看的中秋节网页源码 前言一、设计来源1.1 网站首页界面1.2 登录注册界面1.3 节日由来界面1.4 节日习俗界面1.5 节日文化界面1.6 节日美食界面1.7 节日故事界面1.8 节日民谣界面1.9 联系我们界面 二、效果和源码2.1 动态效果2.2 源代码 源码下载结束语 HTML5实现好看…

ROS通信编程——服务通信

前言 学完了话题通信其实操作流程基本都已经很熟悉了,因此服务通讯的学习就会流畅许多。 服务通信也是ROS中一种极其常用的通信模式,服务通信是基于请求响应模式的,是一种应答机制。也即: 一个节点A向另一个节点B发送请求,B接收…

零基础微信小程序开发——全局配置之tabBar(保姆级教程+超详细)

🎥 作者简介: CSDN\阿里云\腾讯云\华为云开发社区优质创作者,专注分享大数据、Python、数据库、人工智能等领域的优质内容 🌸个人主页: 长风清留杨的博客 🍃形式准则: 无论成就大小,…

2024年12月HarmonyOS应用开发者高级认证全新题库

注意事项:切记在考试之外的设备上打开题库进行搜索,防止切屏三次考试自动结束,题目是乱序,每次考试,选项的顺序都不同,作者已于2024年12月15日又更新了一波题库,题库正确率99%! 新版…

【网络协议】IPv4 地址分配 - 第二部分

前言 在第 1 部分中,我们学习了 IPv4 地址的分配方式,了解了各种类型的 IPv4 地址,并进行了基础的子网划分(Subnetting)。在第 2 部分中,我们将继续学习子网划分,并引入一些新的概念。 【网络…

浙江安吉成新的分布式光伏发电项目应用

摘 要:分布式光伏发电站是指将光伏发电组件安装在用户的建筑物屋顶、空地或其他适合的场地上,利用太阳能进行发电的一种可再生能源利用方式,与传统的大型集中式光伏电站相比,分布式光伏发电具有更灵活的布局、更低的建设成本和更高…

7z 安装与使用指南

文章目录 1. 安装 7z1.1 在 Linux 上安装 7z对于 Debian/Ubuntu 系统:对于 CentOS/RHEL 系统:对于 Fedora 系统:对于 Arch Linux 系统: 1.2 在 Windows 上安装 7z 2. 使用 7z 进行压缩和解压2.1 压缩文件示例:压缩一个…

scanf:数据之舟的摆渡人,静卧输入港湾的诗意守候

大家好啊,我是小象٩(๑ω๑)۶ 我的博客:Xiao Xiangζั͡ޓއއ 很高兴见到大家,希望能够和大家一起交流学习,共同进步。* 这一节我们主要来学习scanf的基本用法,了解scanf返回值,懂得scanf占位符和赋值…

【ROS2】Arduino系列之电机驱动实现

内容学自:赵虚左老师。学后总结! 实现机器人移动的一种策略是:控制系统会先发布预期的车辆速度信息,然后驱动系统订阅到该信息,不断调整电机转速直至达到预期速度,调速过程中还需要时时获取实际速度并反馈…