结构型-代理模式

yizhihongxing

定义

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

es6 中的代理

  es6 的 proxy 就是上面说的代理模式的实现,es6 帮我们在语法层面提供了这个新的api,让我们可以很轻松的就使用了代理模式。

const p = new Proxy(target, handler)
target:要使用 Proxy 包装的目标对象(可以是任何类型的对象,包括原生数组,函数,甚至另一个代理)
handler:一个通常以函数作为属性的对象

 proxy 实例

const handler = {
    get: function(obj, prop) {
        return prop in obj ? obj[prop] : 37;
    }
};

const p = new Proxy({}, handler);
p.a = 1;
p.b = undefined;

console.log(p.a, p.b);      // 1, undefined
console.log('c' in p, p.c); // false, 37

 应用实践-模拟代理模式

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

通常开发中最常见的代理类型:事件代理、虚拟代理、缓存代理、保护代理

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

事件代理

事件代理是代理模式最常见的一种应用方式,它的场景是一个父元素下有多个子元素

<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <meta content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" name="viewport">
  <meta content="yes" name="apple-mobile-web-app-capable">
  <meta content="black" name="apple-mobile-web-app-status-bar-style">
  <meta content="telephone=no,email=no" name="format-detection">
  <meta name="App-Config" content="fullscreen=yes,useHistoryState=yes,transition=yes">
  <meta name="viewport" content="width=device-width, initial-scale=1">
  <title></title>

</head>

<body>
  <p>图片列表--事件代理</p>
  <ul id="ul_wrapper">
    <li>
      1、<img id="img_1" src="" alt="">
    </li>
    <li>
      2、<img id="img_2" src="" alt="">
    </li>
    <li>
      3、<img id="img_3" src="" alt="">
    </li>
    <li>
      4、<img id="img_4" src="" alt="">
    </li>
    <li>
      5、<img id="img_5" src="" alt="">
    </li>
    <li>
      6、<img id="img_6" src="" alt="">
    </li>
    <li>
      7、<img id="img_7" src="" alt="">
    </li>
    <li>
      8、<img id="img_8" src="" alt="">
    </li>
  </ul>
  <script>
      // 自己找个base64,拷贝上来太长了
    let defualtSrc = ``
    let initPage = (function () {
      document.querySelectorAll('img').forEach(item => {
        item.src = defualtSrc
      })
    })();
    document.querySelector('#ul_wrapper').addEventListener('click', function (e) {
      if (e.target.nodeName === 'IMG') {
        alert('图片被点击')
      }
    }, false)
  </script>
</body>

</html>

缓存代理

  缓存代理可以避免重复的计算

<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <meta content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" name="viewport">
  <meta content="yes" name="apple-mobile-web-app-capable">
  <meta content="black" name="apple-mobile-web-app-status-bar-style">
  <meta content="telephone=no,email=no" name="format-detection">
  <meta name="App-Config" content="fullscreen=yes,useHistoryState=yes,transition=yes">
  <meta name="viewport" content="width=device-width, initial-scale=1">
  <title></title>

</head>

<body>
  <p>图片列表--缓存代理</p>
  <button type="button">计算</button>
  <div>
    <label>结果:</label><input type="text">
  </div>
  <script>

    /*
    有效的减少计算;
    工具函数
    */
    const addAll = function (...args) {
      console.log('进行了一次新计算')
      let result = 0
      const len = args.length
      for (let i = 0; i < len; i++) {
        result += args[i]
      }
      return result
    }

    let proxyAddAll = (function () {
      const resultCache = {}
      return function (fn, ...args) {
        const key = args.join('')
        if (resultCache[key]) {
          return resultCache[key]
        }
        return resultCache[key] = fn.apply(this, args)
      }
    })()

    // 123456 参数相同,只是第一次运算的时候,打印了一次进行了一次新计算
    console.log(proxyAddAll(addAll, 1, 2, 3, 4, 5, 6))
    console.log(proxyAddAll(addAll, 1, 2, 3, 4, 5, 6))
    console.log(proxyAddAll(addAll, 1, 2, 3, 4, 5, 6))
    // 1234567 因为是一个全新的参数所以打印了一次进行了一次新计算
    console.log(proxyAddAll(addAll, 1, 2, 3, 4, 5, 7))
  </script>
</body>

</html>

虚拟代理

  图片预加载,预加载主要是为了避免网络不好、或者图片太大时,页面长时间给用户留白的尴尬。原理也很简单创建一个图片实例指向图片真实地址,当完成加载时,把占位图的地址替换成真实的地址,这个时候浏览器会直接从缓存里面拿。

<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <meta content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" name="viewport">
  <meta content="yes" name="apple-mobile-web-app-capable">
  <meta content="black" name="apple-mobile-web-app-status-bar-style">
  <meta content="telephone=no,email=no" name="format-detection">
  <meta name="App-Config" content="fullscreen=yes,useHistoryState=yes,transition=yes">
  <meta name="viewport" content="width=device-width, initial-scale=1">
  <title></title>

</head>

<body>
  <p>图片列表--虚拟代理</p>
  <ul>
    <li>
      1、<img id="img_1" src="" alt="">
    </li>
    <li>
      2、<img id="img_2" src="" alt="">
    </li>
    <li>
      3、<img id="img_3" src="" alt="">
    </li>
    <li>
      4、<img id="img_4" src="" alt="">
    </li>
    <li>
      5、<img id="img_5" src="" alt="">
    </li>
    <li>
      6、<img id="img_6" src="" alt="">
    </li>
    <li>
      7、<img id="img_7" src="" alt="">
    </li>
    <li>
      8、<img id="img_8" src="" alt="">
    </li>
  </ul>
  <script>
    // 替换成你的base64,拷贝上来太长
    let defualtSrc = ""
    let initPage = (function () {
      document.querySelectorAll('img').forEach(item => {
        item.src = defualtSrc
      })
    })();

    // 设置图片地址
    function setImgUrl(dom, src) {
      dom.src = src;
    }

    // 中间的代理图片地址
    function proxyImg(element, url) {
      // 创建一个虚拟Image实例
      const virtualImage = new Image()
      virtualImage.onload = function () {
        setImgUrl(element, url)
      }
      virtualImage.src = url
    }
    function preLoadImg() {
      const urlList = [
        'https://t7.baidu.com/it/u=2621658848,3952322712&fm=193&f=GIF',
        'https://t7.baidu.com/it/u=4080826490,615918710&fm=193&f=GIF',
        'https://t7.baidu.com/it/u=334080491,3307726294&fm=193&f=GIF',
        'https://t7.baidu.com/it/u=3713375227,571533122&fm=193&f=GIF',
        'https://t7.baidu.com/it/u=801209673,1770377204&fm=193&f=GIF',
        'https://t7.baidu.com/it/u=1856946436,1599379154&fm=193&f=GIF',
        'https://t7.baidu.com/it/u=1010739515,2488150950&fm=193&f=GIF',
        'https://t7.baidu.com/it/u=813347183,2158335217&fm=193&f=GIF']
      document.querySelectorAll('img').forEach((element, index) => {
        proxyImg(element, urlList[index])
        element.src = defualtSrc
      })
    }
    setTimeout(() => {
      preLoadImg()
    }, 0.5 * 1000);

    /*
    核心: 有个虚拟的实例去请求地址,拿到之后替换到真实的dom
    */
  </script>
</body>

</html>

保护代理

  可以通过es6 的proxy 的get、set 访问器实现

const handler = {
  get: function(obj, prop) {
      return prop in obj ? obj[prop] : '你不能访问';
  }
};

const p = new Proxy({}, handler);
p.a = 1;
p.b = undefined;

console.log(p.a, p.b);      // 1, undefined
console.log('c' in p, p.c); // false, 你不能访问

具体查看 Proxy
 

小结

 A 不能直接访问 B,A 需要借助一个帮手来访问 B,这个帮手就是代理器,通常开发中最常见的四种代理类型:事件代理、虚拟代理、缓存代理、保护代理;
  1.  事件代理:事件冒泡,代理 DOM
  2. 虚拟代理:通过Image加载图片,代理 DOM
  3. 缓存代理:缓存计算结果,代理函数
  4. 保护代理:get,set保护核心数据,代理对象
 

原文链接:https://www.cnblogs.com/longbensong/p/17255083.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:结构型-代理模式 - Python技术站

(0)
上一篇 2023年4月18日
下一篇 2023年4月18日

相关文章

  • js中replace的用法总结

    以下是详细讲解“js中replace的用法总结”的完整攻略。 replace方法的作用 replace()方法是JavaScript字符串对象的方法。它可以查找并替换字符串中的一些子串。我们可以使用replace方法将一些特殊字符或者字符串转换成其他字符或者字符串。 replace方法的基本用法 string.replace(regexp|substr, n…

    JavaScript 2023年5月28日
    00
  • JavaScript中定时器setTimeout()和setInterval()的用法

    JavaScript中定时器包括setTimeout()和setInterval()两种函数,它们都可以让程序按照指定的时间间隔执行一定的代码。 setTimeout()函数 setTimeout()函数用于在指定的毫秒数后执行一段代码,语法如下: setTimeout(func, delay, param1, param2, …) 其中: func: …

    JavaScript 2023年6月11日
    00
  • JS操作xml对象转换为Json对象示例

    下面是JS操作XML对象转换为JSON对象示例的完整攻略。 什么是XML对象和JSON对象? XML(Extensible Markup Language) 是一种文本格式的标记语言,用于存储和传输数据。XML 文件可作为 Web 文档来读取和呈现。 JSON(JavaScript Object Notation) 是一种轻量级的数据格式,它是一种文本格式的…

    JavaScript 2023年5月27日
    00
  • JS实现运动缓冲效果的封装函数示例

    JS实现运动缓冲效果是前端开发中常见的问题之一。我们可以封装一个函数来实现这个效果,方便快捷地进行运动缓冲效果的实现。 函数封装过程 封装函数涉及到一些关键的概念: 运动开始点 运动结束点 运动距离 运动时间 每一帧的运动距离 运动速度 缓冲系数 基于上述概念,简单描述实现运动缓冲效果的封装函数示例的攻略如下: 确定函数参数 封装函数需要定义函数的参数,以便…

    JavaScript 2023年6月11日
    00
  • js实现键盘自动打字效果

    实现键盘自动打字效果可以分为以下几个步骤: 1. 获取需要自动打印的文本 首先,需要准备需要打印的文本内容。这可以通过在HTML中插入一个元素,并给该元素设置一个文本内容,然后使用JavaScript获取该元素的innerText或innerHTML属性值,就可以得到需要打印的文本。 示例代码 HTML代码: <p id="text&quot…

    JavaScript 2023年5月28日
    00
  • 如何在JS中实现相互转换XML和JSON

    以下是如何在JS中实现相互转换XML和JSON的完整攻略: 将XML转换为JSON格式 在JS中,将XML格式的数据转换为JSON格式的数据,可以使用第三方库:xml2json。该库可以通过npm进行安装: npm install xml2json –save 安装完成后,就可以在JS代码中使用该库进行XML和JSON格式的转换了。下面是一个使用xml2j…

    JavaScript 2023年5月27日
    00
  • 浅析javascript的return语句

    浅析JavaScript的return语句,我们可以从以下几个方面进行阐述: 1. return的作用 在函数内,return关键字的作用是“终止函数执行并返回一个值”。也就是说,当函数执行到return语句时,会立即退出函数并返回一个值,如果没有指定返回值,则返回undefined。 2. return的使用方式 在JavaScript中,return的使…

    JavaScript 2023年6月10日
    00
  • 利用纯css实现缩略图悬停效果实例代码

    下面是利用纯CSS实现缩略图悬停效果的完整攻略。 确定页面布局 首先需要确定页面布局,可以使用HTML和CSS代码实现。通常情况下会使用一个列表来存放所有的缩略图,并设置每个列表项的宽度和高度,如下所示: <ul class="thumbnail-list"> <li> <a href="#&quo…

    JavaScript 2023年5月28日
    00
合作推广
合作推广
分享本页
返回顶部