美文网首页
JS设计模式之外观模式

JS设计模式之外观模式

作者: Splendid飞羽 | 来源:发表于2021-05-16 22:17 被阅读0次

    外观模式 (Facade Pattern)——Facade,又叫门面模式,定义一个将子系统的一组接口集成在一起的高层接口,以提供一个一致的外观。外观模式让外界减少与子系统内多个模块的直接交互,从而减少耦合,让外界可以更轻松地使用子系统。本质是封装交互,简化调用。

    jQuery 源码中的外观模式

    bindReady: function() {
        // ...
        
        // Mozilla, Opera and webkit 支持
        if (document.addEventListener) {
            document.addEventListener('DOMContentLoaded', DOMContentLoaded, false)
            
            // A fallback to window.onload, that will always work
            window.addEventListener('load', jQuery.ready, false)
            
            // 如果使用了 IE 的事件绑定形式
        } else if (document.attachEvent) {
            document.attachEvent('onreadystatechange', DOMContentLoaded)
            
            // A fallback to window.onload, that will always work
            window.attachEvent('onload', jQuery.ready)
        }
        
        // ...
    }
    

    通过这个方法,jQuery 帮我们将不同浏览器下的不同绑定形式隐藏起来,从而简化了使用。
    bindReady 方法的源码参见 Github 链接 jquery/src/core.js

    除了屏蔽浏览器兼容性问题之外,jQuery 还有其他的一些其他外观模式的应用:

    比如修改 css 的时候可以 ('p').css('color', 'red') ,也可以('p').css('width', 100),对不同样式的操作被封装到同一个外观方法中,极大地方便了使用,对不同样式的特殊处理(比如设置 width 的时候不用加 px)也一同被封装了起来。
    源码参见 Github 链接 jquery/src/css.js
    再比如 jQuery 的 ajax 的 API .ajax(url [, settings]),当我们在设置以 JSONP的形式发送请求的时候,只要传入 dataType: 'jsonp' 设置,jQuery会进行一些额外操作帮我们启动JSONP流程,并不需要使用者手动添加代码,这些都被封装在.ajax()` 这个外观方法中了。
    源码参见 Github 链接 jquery/src/ajax/jsonp.js

    Axios 源码中的外观模式

    Axios 可以使用在不同环境中,那么在不同环境中发送 HTTP 请求的时候会使用不同环境中的特有模块,Axios 这里是使用外观模式来解决这个问题的:

    function getDefaultAdapter() {
      // ...
    
      if (typeof process !== 'undefined' && Object.prototype.toString.call(process) === '[object process]') {
        // Nodejs 中使用 HTTP adapter
        adapter = require('./adapters/http');
      } else if (typeof XMLHttpRequest !== 'undefined') {
        // 浏览器使用 XHR adapter
        adapter = require('./adapters/xhr');
      }
      
      // ...
    }
    

    getDefaultAdapter 方法源码参见 Github 链接 axios/lib/defaults.js

    1、外观模式的实例:函数参数重载

    有一种情况,比如某个函数有多个参数,其中一个参数可以传递也可以不传递,你当然可以直接弄两个接口,但是使用函数参数重载的方式,可以让使用者获得更大的自由度,让两个使用上基本类似的方法获得统一的外观。

    function domBindEvent(nodes, type, selector, fn) {
        if (fn === undefined) {
            fn = selector
            selector = null
        }
        // ... 剩下相关逻辑
    }
    
    domBindEvent(nodes, 'click', '#div1', fn)
    domBindEvent(nodes, 'click', fn)
    

    再比如vue源码中的createElement就用到了

    export function createElement(
      context,
      tag,
      data,
      children,
      normalizationType,
      alwaysNormalize
    ) {
        if (Array.isArray(data) || isPrimitive(data)) {     // 参数的重载
            normalizationType = children
            children = data
            data = undefined
        }
        
        // ...
    }
    
    2、解决浏览器兼容性

    外观模式经常被用于 JavaScript 的库中,封装一些接口用于兼容多浏览器,让我们可以间接调用我们封装的外观,从而屏蔽了浏览器差异,便于使用。

    function addEvent(element, type, fn) {
        if (element.addEventListener) {      // 支持 DOM2 级事件处理方法的浏览器
            element.addEventListener(type, fn, false)
        } else if (element.attachEvent) {    // 不支持 DOM2 级但支持 attachEvent
            element.attachEvent('on' + type, fn)
        } else {
            element['on' + type] = fn        // 都不支持的浏览器
        }
    }
    
    var myInput = document.getElementById('myinput')
    
    addEvent(myInput, 'click', function() {
        console.log('绑定 click 事件')
    })
    
    总结:
    外观模式的优点:

    访问者不需要再了解子系统内部模块的功能,而只需和外观交互即可,使得访问者对子系统的使用变得简单,符合最少知识原则,增强了可移植性和可读性;
    减少了与子系统模块的直接引用,实现了访问者与子系统中模块之间的松耦合,增加了可维护性和可扩展性;
    通过合理使用外观模式,可以帮助我们更好地划分系统访问层次,比如把需要暴露给外部的功能集中到外观中,这样既方便访问者使用,也很好地隐藏了内部的细节,提升了安全性;

    外观模式的缺点:

    不符合开闭原则,对修改关闭,对扩展开放,如果外观模块出错,那么只能通过修改的方式来解决问题,因为外观模块是子系统的唯一出口;
    不需要或不合理的使用外观会让人迷惑,过犹不及;

    相关文章

      网友评论

          本文标题:JS设计模式之外观模式

          本文链接:https://www.haomeiwen.com/subject/jkarjltx.html