当前位置: 首页 > news >正文

设计模式之-代理模式

1.代理实现懒加载1

varmyImage=(function(){varimg=document.createElement('img');document.body.appendChild(img);return{setSrc:function(src){img.src=src;}}})();varproxyImage=(function(){// 创建预加载图片varimg=newImage();img.onload=function(){// 预加载完成后,设置真实图片myImage.setSrc(this.src)}return{setSrc:function(src){// 先显示loading图片myImage.setSrc('./loading.png');img.src=src;}}})();// 使用示例proxyImage.setSrc('https://fastly.picsum.photos/id/784/200/300.jpg?hmac=LIWlcHgxQH79XHKNji8Jin_KakntjYyd9VXyckNYFbE')

代理实现懒加载2,class的写法

classMyImage{constructor(){this.img=document.createElement('img');document.body.appendChild(this.img);}setSrc(src){this.img.src=src;}}classProxyImage{constructor(realImage){this.img=realImage;this.cacheImage=null;// 预加载的图片对象}setSrc(src){// 先显示loading图片this.img.setSrc('./loading.png');// 创建预加载图片this.cacheImage=newImage();this.cacheImage.onload=()=>{// 预加载完成后,设置真实图片this.img.setSrc(src);};this.cacheImage.src=src;}}// 使用示例constimg=newMyImage();constproxyImg=newProxyImage(img);proxyImg.setSrc('https://fastly.picsum.photos/id/784/200/300.jpg?hmac=LIWlcHgxQH79XHKNji8Jin_KakntjYyd9VXyckNYFbE')

代理引发的思考:也许有人会有疑惑,不过是实现一个小小的图片懒加载,即使不引用任何模式也能办到,那么引入代理模式的好处究竟在哪里呢?下面我们先抛开代理,编写一个更常见的图片预加载函数,不使用代理的与加载图片函数实现如下:

varmyImage=(function(){varimgNode=document.createElement('img');document.body.appendChild(imgNode);varimg=newImage();img.onload=function(){// 预加载完成后,设置真实图片imgNode.src=img.src}return{setSrc:function(src){imgNode.src='./loading.png';img.src=src}}})();// 使用示例myImage.setSrc('https://fastly.picsum.photos/id/784/200/300.jpg?hmac=LIWlcHgxQH79XHKNji8Jin_KakntjYyd9VXyckNYFbE')

代理的意义

1.为了说明代理的意义,下面我们引入一个面向对象设计的原则–单一职责原则。
2.单一职责原则指的是,就是一个类(通常也包括对象和函数等)而言,应该仅有一个引起他变化的原因。如果一个对象承担了多项职责,就意味着这个对象将变得巨大,引起他变化的原因可能有多个。面向对象设计鼓励将行为分布到细粒度的对象当中,如果一个对象承担的职责过多,等于把这些职责耦合到了一起,这种耦合会导致脆弱和低内聚的设计,当变化发生时,设计可能会遭到意外的破坏。
3.职责被定义为“引起变化的原因”。上面代码中的MyImage对象除了负责给img节点设置src外,还要负责预加载图片。我们在处理其中一个职责的时候,有可能因为其强耦合性影响另外一个职责的实现。
4.另外,在面向对象的程序设计中,大多数情况下,若违反其他任何原则,同时将违反开放-封闭原则。如果我们只是从网络上获取一些体积很小的图片,或者5年后网速快到根本不在需要预加载了,我们可能希望把预加载图片的这段代码从MyImage对象里删掉。这时候就不得不改动MyImage对象了。
5.实际上,我们需要的只是给img节点设置src,预加载图片只是一个锦上添花的功能。如果能把这个操作放到另一个对象里面,自然是一个很好的方法。于是代理的作用就体现出来了,代理负责预加载图片,预加载的操作完成后,把请求重新交给本体MyImage.
6.纵观整个程序,我们并没有改变或者增加MyImage的接口,但是通过代理对象,实际上给系统添加了新的行为。这是符合开放-封闭原则的。给img节点设置src和图片预加载这两个功能,被隔离在两个对象里,他们可以各自变化而不影响对方。何况就算有一天我们不再需要预加载,那么只需要改成请求本体而不是请求代理对象即可。

代理和本体接口的一致性

1.如果有一天我们不再需要预加载了,那么就不在需要代理对象,可以选择直接请求本体。其中关键是代理对象和本地都对外提供了setSrc方法,在客户看来,代理对象和本体是一致的,代理接手请求的过程对于用户来说是透明的,用户并不清楚代理和本体的区别,那么这么做有两个好处
(1)用户可以放心的请求代理,他只关心是否能得到想要的结果
(2)在任何使用本体的地方都可以替换成使用代理
2.在Java等语言中,代理和本体都需要显式地实现同一个接口,一方面接口保证了他们会拥有同样的方法,另一方面,面向接口编程迎合依赖倒置原则,通过接口进行向上转型,从而避开编译器的类型检查,代理和本体将来可以被替换使用
3.在javascript这种动态类型语言中,我们有时候通过鸭子类型来检测代理和本体是否都实现setSrc方法,另外大多数时候甚至干脆不做检测,全部依赖程序员的自觉性,这对于程序的健壮性是有影响的。不过对于一门快速开发的脚本语言,这些影响还是在可以接受的范围内,而且我们也习惯了没有接口的世界。
4.另外值得一提的是,如果代理对象和本体对象都为一个函数(函数也是对象),而函数必然都能被执行,则可以认为它们具有一致的“接口”,代码如下:

varmyImage=(function(){varimg=document.createElement('img');document.body.appendChild(img);returnfunction(src){img.src=src;}})();varproxyImage=(function(){// 创建预加载图片varimg=newImage();img.onload=function(){// 预加载完成后,设置真实图片myImage(this.src)}returnfunction(src){// 先显示loading图片myImage('./loading.png');img.src=src;}})();// 使用示例proxyImage('https://fastly.picsum.photos/id/784/200/300.jpg?hmac=LIWlcHgxQH79XHKNji8Jin_KakntjYyd9VXyckNYFbE')

2.ts版本的demo

interfaceIService{operation():void;}// 提供真实服务的类classServiceimplementsIService{operation():void{console.log("Service operation called");}}classMyProxyimplementsIService{// 代理类中包含一个真实服务的引用privaterealService:Service;constructor(se:Service){this.realService=se;}// 对外界的访问进行一个过滤checkAccess():boolean{// 这边可以进行一些访问控制的操作// ...returntrue;}// 实现接口中定义的方法operation():void{// 这边需要对外界的访问进行一个过滤if(this.checkAccess()){// 经过访问控制后,可以调用真实服务的方法this.realService.operation();}}}// 使用constrealService=newService();constproxy=newMyProxy(realService);// 外界要访问服务的时候,是和代理类打交道的proxy.operation();

3.前端中的代理模式-- ES6新增的Proxy构造方法

// 创建一个要被代理的对象// 这是真实的对象consttarget={name:"John",age:30,};// 定义代理的行为// 代理对象就会对外界的访问需求进行过滤consthandler={// 拦截对象属性的读取get(target,prop,receiver){console.log(`[GET]:${prop}`);returnReflect.get(target,prop,receiver);},// 拦截对象属性的设置set(target,prop,value,receiver){console.log(`[SET]:${prop}=${value}`);returnReflect.set(target,prop,value,receiver);},};// 使用Proxy构造函数创建代理对象// new Proxy 会返回一个对象,该对象就是针对真实对象的代理对象constproxy=newProxy(target,handler);// 测试代理的行为// 之后就通过代理对象来访问真实对象的成员console.log(proxy.name);// 输出:[GET]: name 以及 Johnproxy.age=31;// 输出:[SET]: age = 31

非原创,来源渡一谢杰老师和javascript设计模式与开发实践 作者曾探,简单记录,一起学起来吧

http://www.cnnetsun.cn/news/136808.html

相关文章:

  • SIEMENS 6SL3210-1PE33-0CL0 变频器
  • 软件测试常用的7种方法,最后一个是升职加薪关键!(零基础小白转行IT互联网高效进阶)
  • 【RTOS】EasyLog的移植与使用
  • 在数据库里玩“平行宇宙”:MatrixOne Data Branch 让数据也拥有Git 的分支/合并/对比/回滚(含跨集群同步)
  • 基于单片机的全自动洗衣机系统的设计
  • 5.6 模型部署与智能体集成实战
  • 基于单片机的球赛计分牌的设计
  • ArcGIS Pro 从入门到实战基础篇(10):地图菜单
  • Kotaemon与Redis/Memcached集成:构建高速缓存层
  • 【鸿蒙三方库编译】lycium_plusplus(lycium++)高效完成鸿蒙C/C++编译
  • 2025年度GEO服务商权威甄选指南:技术深度与商业价值的双重考量
  • 收藏备用!Java程序员转AI大模型:从技术沉淀到AI爆发的进阶之路
  • Python 爬虫实战:Session 会话维持爬取需登录内容
  • 基于移相全桥变换器的电池充电仿真模型,采用电压电流双闭环PI控制。 电池先经历CC模式而后进入...
  • 基于COMSOL模拟的水力压裂技术研究:固体力学与达西定理的应用
  • Redis 性能调优(二)
  • Doris 性能调优实践指南(可直接落地)
  • presum|二分try+滑窗cnt
  • Web自动化测试:Unittest单元测试框架
  • Apache2最佳实践
  • 实力派,也可以是偶像派
  • 基于单片机的多功能万年历
  • AI搜索时代:技术演进、产业分化与深度变革
  • SGMICRO圣邦微 SGM2019-2.5YC5G/TR SC70-5 线性稳压器(LDO)
  • 一文搞懂 低功耗蓝牙BLE 中的 ATT、GATT、MTU 与 20 字节限制
  • 别让“大锅饭”逼走你的Top Sales:揭秘薪酬误差的副作用
  • 27827828
  • 12.17 vue递归组件
  • QtScrcpy高刷投屏优化指南:告别卡顿,享受流畅体验
  • 终极移动端Windows应用运行指南:从零到流畅体验