翼度科技»论坛 编程开发 JavaScript 查看内容

结构型-代理模式

2

主题

2

帖子

6

积分

新手上路

Rank: 1

积分
6
定义

  代理是一个中间者的角色,如生活中的中介,出于种种考虑/限制,一个对象不能直接访问另一个对象,需要一个第三者(中间代理)牵线搭桥从而间接达到访问目的,这样的就是代理模式。

es6 中的代理


  es6 的 proxy 就是上面说的代理模式的实现,es6 帮我们在语法层面提供了这个新的api,让我们可以很轻松的就使用了代理模式。
  1. const p = new Proxy(target, handler)
  2. target:要使用 Proxy 包装的目标对象(可以是任何类型的对象,包括原生数组,函数,甚至另一个代理)
  3. handler:一个通常以函数作为属性的对象
复制代码
 proxy 实例
  1. const handler = {
  2.     get: function(obj, prop) {
  3.         return prop in obj ? obj[prop] : 37;
  4.     }
  5. };
  6. const p = new Proxy({}, handler);
  7. p.a = 1;
  8. p.b = undefined;
  9. console.log(p.a, p.b);      // 1, undefined
  10. console.log('c' in p, p.c); // false, 37
复制代码
 应用实践-模拟代理模式


  代理模式的应用非常常见,既可以是为了加强控制、拓展功能、提高性能,也可以仅仅是为了优化我们的代码结构、实现功能的解耦。无论是出于什么目的,这种模式的套路就只有一个—— A 不能直接访问 B,A 需要借助一个帮手来访问 B,这个帮手就是代理器。需要这种代理器的就是代理模式的应用场景。
通常开发中最常见的代理类型:事件代理、虚拟代理、缓存代理、保护代理

  • 事件代理:代理 DOM
  • 虚拟代理:代理 DOM
  • 缓存代理:代理函数
  • 保护代理:代理对象
事件代理

事件代理是代理模式最常见的一种应用方式,它的场景是一个父元素下有多个子元素
  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4.   <meta charset="UTF-8">
  5.   <meta content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" name="viewport">
  6.   <meta content="yes" name="apple-mobile-web-app-capable">
  7.   <meta content="black" name="apple-mobile-web-app-status-bar-style">
  8.   <meta content="telephone=no,email=no" name="format-detection">
  9.   <meta name="App-Config" content="fullscreen=yes,useHistoryState=yes,transition=yes">
  10.   <meta name="viewport" content="width=device-width, initial-scale=1">
  11.   <title></title>
  12. </head>
  13. <body>
  14.   <p>图片列表--事件代理</p>
  15.   <ul id="ul_wrapper">
  16.     <li>
  17.       1、<img id="img_1" src="" alt="">
  18.     </li>
  19.     <li>
  20.       2、<img id="img_2" src="" alt="">
  21.     </li>
  22.     <li>
  23.       3、<img id="img_3" src="" alt="">
  24.     </li>
  25.     <li>
  26.       4、<img id="img_4" src="" alt="">
  27.     </li>
  28.     <li>
  29.       5、<img id="img_5" src="" alt="">
  30.     </li>
  31.     <li>
  32.       6、<img id="img_6" src="" alt="">
  33.     </li>
  34.     <li>
  35.       7、<img id="img_7" src="" alt="">
  36.     </li>
  37.     <li>
  38.       8、<img id="img_8" src="" alt="">
  39.     </li>
  40.   </ul>
  41.   
  42. </body>
  43. </html>
复制代码
缓存代理

  缓存代理可以避免重复的计算
  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4.   <meta charset="UTF-8">
  5.   <meta content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" name="viewport">
  6.   <meta content="yes" name="apple-mobile-web-app-capable">
  7.   <meta content="black" name="apple-mobile-web-app-status-bar-style">
  8.   <meta content="telephone=no,email=no" name="format-detection">
  9.   <meta name="App-Config" content="fullscreen=yes,useHistoryState=yes,transition=yes">
  10.   <meta name="viewport" content="width=device-width, initial-scale=1">
  11.   <title></title>
  12. </head>
  13. <body>
  14.   <p>图片列表--缓存代理</p>
  15.   <button type="button">计算</button>
  16.   
  17.     <label>结果:</label><input type="text">
  18.   
  19.   
  20. </body>
  21. </html>
复制代码
虚拟代理

  图片预加载,预加载主要是为了避免网络不好、或者图片太大时,页面长时间给用户留白的尴尬。原理也很简单创建一个图片实例指向图片真实地址,当完成加载时,把占位图的地址替换成真实的地址,这个时候浏览器会直接从缓存里面拿。
  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4.   <meta charset="UTF-8">
  5.   <meta content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" name="viewport">
  6.   <meta content="yes" name="apple-mobile-web-app-capable">
  7.   <meta content="black" name="apple-mobile-web-app-status-bar-style">
  8.   <meta content="telephone=no,email=no" name="format-detection">
  9.   <meta name="App-Config" content="fullscreen=yes,useHistoryState=yes,transition=yes">
  10.   <meta name="viewport" content="width=device-width, initial-scale=1">
  11.   <title></title>
  12. </head>
  13. <body>
  14.   <p>图片列表--虚拟代理</p>
  15.   <ul>
  16.     <li>
  17.       1、<img id="img_1" src="" alt="">
  18.     </li>
  19.     <li>
  20.       2、<img id="img_2" src="" alt="">
  21.     </li>
  22.     <li>
  23.       3、<img id="img_3" src="" alt="">
  24.     </li>
  25.     <li>
  26.       4、<img id="img_4" src="" alt="">
  27.     </li>
  28.     <li>
  29.       5、<img id="img_5" src="" alt="">
  30.     </li>
  31.     <li>
  32.       6、<img id="img_6" src="" alt="">
  33.     </li>
  34.     <li>
  35.       7、<img id="img_7" src="" alt="">
  36.     </li>
  37.     <li>
  38.       8、<img id="img_8" src="" alt="">
  39.     </li>
  40.   </ul>
  41.   
  42. </body>
  43. </html>
复制代码
保护代理

  可以通过es6 的proxy 的get、set 访问器实现
  1. const handler = {
  2.   get: function(obj, prop) {
  3.       return prop in obj ? obj[prop] : '你不能访问';
  4.   }
  5. };
  6. const p = new Proxy({}, handler);
  7. p.a = 1;
  8. p.b = undefined;
  9. console.log(p.a, p.b);      // 1, undefined
  10. console.log('c' in p, p.c); // false, 你不能访问
复制代码
具体查看 proxyProxy 小结

  A 不能直接访问 B,A 需要借助一个帮手来访问 B,这个帮手就是代理器,通常开发中最常见的四种代理类型:事件代理、虚拟代理、缓存代理、保护代理;

  •  事件代理:事件冒泡,代理 DOM
  • 虚拟代理:通过Image加载图片,代理 DOM
  • 缓存代理:缓存计算结果,代理函数
  • 保护代理:get,set保护核心数据,代理对象
 
来源:https://www.cnblogs.com/longbensong/p/17255083.html
免责声明:由于采集信息均来自互联网,如果侵犯了您的权益,请联系我们【E-Mail:cb@itdo.tech】 我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x

举报 回复 使用道具