深入理解JavaScript系列(31):设计模式之代理模式详解


Posted in Javascript onMarch 03, 2015

介绍

代理,顾名思义就是帮助别人做事,GoF对代理模式的定义如下:

代理模式(Proxy),为其他对象提供一种代理以控制对这个对象的访问。

代理模式使得代理对象控制具体对象的引用。代理几乎可以是任何对象:文件,资源,内存中的对象,或者是一些难以复制的东西。

正文

我们来举一个简单的例子,假如dudu要送酸奶小妹玫瑰花,却不知道她的联系方式或者不好意思,想委托大叔去送这些玫瑰,那大叔就是个代理(其实挺好的,可以扣几朵给媳妇),那我们如何来做呢?

// 先声明美女对象

var girl = function (name) {

    this.name = name;

};
// 这是dudu

var dudu = function (girl) {

    this.girl = girl;

    this.sendGift = function (gift) {

        alert("Hi " + girl.name + ", dudu送你一个礼物:" + gift);

    }

};
// 大叔是代理

var proxyTom = function (girl) {

    this.girl = girl;

    this.sendGift = function (gift) {

        (new dudu(girl)).sendGift(gift); // 替dudu送花咯

    }

};

调用方式就非常简单了:

var proxy = new proxyTom(new girl("酸奶小妹"));

proxy.sendGift("999朵玫瑰");

实战一把

通过上面的代码,相信大家对代理模式已经非常清楚了,我们来实战下:我们有一个简单的播放列表,需要在点击单个连接(或者全选)的时候在该连接下方显示视频曲介绍以及play按钮,点击play按钮的时候播放视频,列表结构如下:

<h1>Dave Matthews vids</h1>

<p><span id="toggle-all">全选/反选</span></p>

<ol id="vids">

  <li><input type="checkbox" checked><a href="http://new.music.yahoo.com/videos/--2158073">Gravedigger</a></li>

  <li><input type="checkbox" checked><a href="http://new.music.yahoo.com/videos/--4472739">Save Me</a></li>

  <li><input type="checkbox" checked><a href="http://new.music.yahoo.com/videos/--45286339">Crush</a></li>

  <li><input type="checkbox" checked><a href="http://new.music.yahoo.com/videos/--2144530">Don't Drink The Water</a></li>

  <li><input type="checkbox" checked><a href="http://new.music.yahoo.com/videos/--217241800">Funny the Way It Is</a></li>

  <li><input type="checkbox" checked><a href="http://new.music.yahoo.com/videos/--2144532">What Would You Say</a>

</li>

</ol>

我们先来分析如下,首先我们不仅要监控a连接的点击事件,还要监控“全选/反选”的点击事件,然后请求服务器查询视频信息,组装HTML信息显示在li元素的最后位置上,效果如下:

深入理解JavaScript系列(31):设计模式之代理模式详解

然后再监控play连接的点击事件,点击以后开始播放,效果如下:

深入理解JavaScript系列(31):设计模式之代理模式详解

好了,开始,没有jQuery,我们自定义一个选择器:

var $ = function (id) {

    return document.getElementById(id);

};

由于Yahoo的json服务提供了callback参数,所以我们传入我们自定义的callback以便来接受数据,具体查询字符串拼装代码如下:
var http = {

    makeRequest: function (ids, callback) {

        var url = 'http://query.yahooapis.com/v1/public/yql?q=',

            sql = 'select * from music.video.id where ids IN ("%ID%")',

            format = "format=json",

            handler = "callback=" + callback,

            script = document.createElement('script');
            sql = sql.replace('%ID%', ids.join('","'));

            sql = encodeURIComponent(sql);
            url += sql + '&' + format + '&' + handler;

            script.src = url;
        document.body.appendChild(script);

    }

};

代理对象如下:

var proxy = {

    ids: [],

    delay: 50,

    timeout: null,

    callback: null,

    context: null,

    // 设置请求的id和callback以便在播放的时候触发回调

    makeRequest: function (id, callback, context) {
        // 添加到队列dd to the queue

        this.ids.push(id);
        this.callback = callback;

        this.context = context;
        // 设置timeout

        if (!this.timeout) {

            this.timeout = setTimeout(function () {

                proxy.flush();

            }, this.delay);

        }

    },

    // 触发请求,使用代理职责调用了http.makeRequest

    flush: function () {

        // proxy.handler为请求yahoo时的callback

        http.makeRequest(this.ids, 'proxy.handler'); 

        // 请求数据以后,紧接着执行proxy.handler方法(里面有另一个设置的callback)

        

        // 清楚timeout和队列

        this.timeout = null;

        this.ids = [];
    },

    handler: function (data) {

        var i, max;
        // 单个视频的callback调用

        if (parseInt(data.query.count, 10) === 1) {

            proxy.callback.call(proxy.context, data.query.results.Video);

            return;

        }
        // 多个视频的callback调用

        for (i = 0, max = data.query.results.Video.length; i < max; i += 1) {

            proxy.callback.call(proxy.context, data.query.results.Video[i]);

        }

    }

};

视频处理模块主要有3种子功能:获取信息、展示信息、播放视频:

var videos = {

    // 初始化播放器代码,开始播放

    getPlayer: function (id) {

        return '' +

            '<object width="400" height="255" id="uvp_fop" allowFullScreen="true">' +

            '<param name="movie" value="http://d.yimg.com/m/up/fop/embedflv/swf/fop.swf"\/>' +

            '<param name="flashVars" value="id=v' + id + '&eID=1301797&lang=us&enableFullScreen=0&shareEnable=1"\/>' +

            '<param name="wmode" value="transparent"\/>' +

            '<embed ' +

            'height="255" ' +

            'width="400" ' +

            'id="uvp_fop" ' +

            'allowFullScreen="true" ' +

            'src="http://d.yimg.com/m/up/fop/embedflv/swf/fop.swf" ' +

            'type="application/x-shockwave-flash" ' +

            'flashvars="id=v' + id + '&eID=1301797&lang=us&ympsc=4195329&enableFullScreen=1&shareEnable=1"' +

            '\/>' +

            '<\/object>';

                },

    // 拼接信息显示内容,然后在append到li的底部里显示

    updateList: function (data) {

        var id,

            html = '',

            info;
        if (data.query) {

            data = data.query.results.Video;

        }

        id = data.id;

        html += '<img src="' + data.Image[0].url + '" width="50" \/>';

        html += '<h2>' + data.title + '<\/h2>';

        html += '<p>' + data.copyrightYear + ', ' + data.label + '<\/p>';

        if (data.Album) {

            html += '<p>Album: ' + data.Album.Release.title + ', ' + data.Album.Release.releaseYear + '<br \/>';

        }

        html += '<p><a class="play" href="http://new.music.yahoo.com/videos/--' + id + '">» play<\/a><\/p>';

        info = document.createElement('div');

        info.id = "info" + id;

        info.innerHTML = html;

        $('v' + id).appendChild(info);

    },

    // 获取信息并显示

    getInfo: function (id) {

        var info = $('info' + id);
        if (!info) {

            proxy.makeRequest(id, videos.updateList, videos); //执行代理职责,并传入videos.updateList回调函数

            return;

        }
        if (info.style.display === "none") {

            info.style.display = '';

        } else {

            info.style.display = 'none';

        }

    }

};

现在可以处理点击事件的代码了,由于有很多a连接,如果每个连接都绑定事件的话,显然性能会有问题,所以我们将事件绑定在<ol>元素上,然后检测点击的是否是a连接,如果是说明我们点击的是视频地址,然后就可以播放了:

$('vids').onclick = function (e) {

    var src, id;
    e = e || window.event;

    src = e.target || e.srcElement;
    // 不是连接的话就不继续处理了

    if (src.nodeName.toUpperCase() !== "A") {

        return;

    }

    //停止冒泡

    if (typeof e.preventDefault === "function") {

        e.preventDefault();

    }

    e.returnValue = false;
    id = src.href.split('--')[1];
    //如果点击的是已经生产的视频信息区域的连接play,就开始播放

    // 然后return不继续了

    if (src.className === "play") {

        src.parentNode.innerHTML = videos.getPlayer(id);

        return;

    }

        

    src.parentNode.id = "v" + id;

    videos.getInfo(id); // 这个才是第一次点击的时候显示视频信息的处理代码

};

全选反选的代码大同小异,我们就不解释了:

$('toggle-all').onclick = function (e) {
    var hrefs, i, max, id;
    hrefs = $('vids').getElementsByTagName('a');

    for (i = 0, max = hrefs.length; i < max; i += 1) {

        // 忽略play连接

        if (hrefs[i].className === "play") {

            continue;

        }

        // 忽略没有选择的项

        if (!hrefs[i].parentNode.firstChild.checked) {

            continue;

        }
        id = hrefs[i].href.split('--')[1];

        hrefs[i].parentNode.id = "v" + id;

        videos.getInfo(id);

    }

};

总结

代理模式一般适用于如下场合:

1.远程代理,也就是为了一个对象在不同的地址空间提供局部代表,这样可以隐藏一个对象存在于不同地址空间的事实,就像web service里的代理类一样。
2.虚拟代理,根据需要创建开销很大的对象,通过它来存放实例化需要很长时间的真实对象,比如浏览器的渲染的时候先显示问题,而图片可以慢慢显示(就是通过虚拟代理代替了真实的图片,此时虚拟代理保存了真实图片的路径和尺寸。
3.安全代理,用来控制真实对象访问时的权限,一般用于对象应该有不同的访问权限。
4.智能指引,只当调用真实的对象时,代理处理另外一些事情。例如C#里的垃圾回收,使用对象的时候会有引用次数,如果对象没有引用了,GC就可以回收它了。

参考:《大话设计模式》

Javascript 相关文章推荐
JS宝典学习笔记(下)
Jan 10 Javascript
关于js类的定义
Jun 28 Javascript
基于jQuery的前端数据通用验证库
Aug 08 Javascript
javascript设计模式之中介者模式Mediator
Dec 30 Javascript
Javascript核心读书有感之词法结构
Feb 01 Javascript
JavaScript获取当前时间向前推三个月的方法示例
Feb 04 Javascript
详解Vue结合后台的列表增删改案例
Aug 21 Javascript
微信小程序Getuserinfo解决方案图解
Aug 24 Javascript
vue实现下拉加载其实没那么复杂
Aug 13 Javascript
基于vue--key值的特殊用处详解
Jul 31 Javascript
JavaScript代码简化技巧实例解析
Sep 09 Javascript
Vue+TypeScript中处理computed方式
Apr 02 Vue.js
深入理解JavaScript系列(30):设计模式之外观模式详解
Mar 03 #Javascript
深入理解JavaScript系列(29):设计模式之装饰者模式详解
Mar 03 #Javascript
jQuery对象与DOM对象之间的相互转换
Mar 03 #Javascript
深入理解JavaScript系列(28):设计模式之工厂模式详解
Mar 03 #Javascript
JS运动基础框架实例分析
Mar 03 #Javascript
jQuery DOM插入节点操作指南
Mar 03 #Javascript
JS运动框架之分享侧边栏动画实例
Mar 03 #Javascript
You might like
php array_walk() 数组函数
2011/07/12 PHP
php实现专业获取网站SEO信息类实例
2015/04/02 PHP
Zend Framework数据库操作技巧总结
2017/02/18 PHP
PHP call_user_func和call_user_func_array函数的简单理解与应用分析
2019/11/25 PHP
javascript判断css3动画结束 css3动画结束的回调函数
2015/03/10 Javascript
js原型链与继承解析(初体验)
2016/05/09 Javascript
JavaScript面向对象编写购物车功能
2016/08/19 Javascript
jQuery中DOM节点的删除方法总结(超全面)
2017/01/22 Javascript
jquery easyui dataGrid动态改变排序字段名的方法
2017/03/02 Javascript
nodejs实现邮件发送服务实例分享
2017/03/29 NodeJs
利用Node.js批量抓取高清妹子图片实例教程
2018/08/02 Javascript
原生js检测页面加载完毕的实例
2018/09/11 Javascript
JavaScript刷新页面的几种方法总结
2019/03/28 Javascript
Fetch超时设置与终止请求详解
2019/05/18 Javascript
Element Alert警告的具体使用方法
2020/07/27 Javascript
Python 常用的安装Module方式汇总
2017/05/06 Python
Python通过Django实现用户注册和邮箱验证功能代码
2017/12/11 Python
详解Python安装scrapy的正确姿势
2018/06/26 Python
基于python实现文件加密功能
2020/01/06 Python
基于python检查SSL证书到期情况代码实例
2020/04/04 Python
python用什么编辑器进行项目开发
2020/06/17 Python
关于Kotlin中SAM转换的那些事
2020/09/15 Python
python time()的实例用法
2020/11/03 Python
htnl5利用svg页面高斯模糊的方法
2018/07/20 HTML / CSS
全球最大的在线橄榄球商店:Lovell Rugby
2018/05/20 全球购物
应届生高等护理求职信
2013/10/12 职场文书
表决心的诗句大全
2014/03/11 职场文书
环境工程专业自荐信范文
2014/03/18 职场文书
运输服务质量承诺书
2014/03/27 职场文书
法制宣传标语
2014/06/23 职场文书
学校机关党总支领导班子整改工作方案
2014/10/26 职场文书
事业单位个人总结
2015/02/12 职场文书
高中体育课教学反思
2016/02/16 职场文书
初一数学教学反思
2016/02/17 职场文书
python 实现定时任务的四种方式
2021/04/01 Python
MySQL创建定时任务
2022/01/22 MySQL