您好,欢迎访问一九零五行业门户网

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

介绍
代理,顾名思义就是帮助别人做事,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按钮的时候播放视频,列表结构如下:
复制代码 代码如下:
dave matthews vids
全选/反选
gravedigger
  save me
  crush
  don't drink the water
  funny the way it is
  what would you say
我们先来分析如下,首先我们不仅要监控a连接的点击事件,还要监控“全选/反选”的点击事件,然后请求服务器查询视频信息,组装html信息显示在li元素的最后位置上,效果如下:
然后再监控play连接的点击事件,点击以后开始播放,效果如下:
好了,开始,没有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             proxy.callback.call(proxy.context, data.query.results.video[i]);
        }
    }
};
视频处理模块主要有3种子功能:获取信息、展示信息、播放视频:
复制代码 代码如下:
var videos = {
    // 初始化播放器代码,开始播放
    getplayer: function (id) {
        return '' +
            '' +
            '' +
            '' +
            '' +
            '            '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' +
            '\/>' +
            '';
                },
    // 拼接信息显示内容,然后在append到li的底部里显示
    updatelist: function (data) {
        var id,
            html = '',
            info;        if (data.query) {
            data = data.query.results.video;
        }
        id = data.id;
        html += '';
        html += '
' + data.title + '';
        html += '' + data.copyrightyear + ', ' + data.label + '';
        if (data.album) {
            html += '
album: ' + data.album.release.title + ', ' + data.album.release.releaseyear + '
';
        }
        html += '
» play';
        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连接,如果每个连接都绑定事件的话,显然性能会有问题,所以我们将事件绑定在
元素上,然后检测点击的是否是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         // 忽略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就可以回收它了。
参考:《大话设计模式》
其它类似信息

推荐信息