javascript设计模式:方法的链式调用

2016-02-20 01:07 8 1 收藏

想不想get新技能酷炫一下,今天图老师小编就跟大家分享个简单的javascript设计模式:方法的链式调用教程,一起来看看吧!超容易上手~

【 tulaoshi.com - Web开发 】

在javascript中,方法的链式调用很流行,使用jQuery的朋友对此一定深有体会。在《javascript设计模式》中对这种方法作了比较详细的描述,实现方法的链式调用,只须让在原型中定义的方法都返回调用这些方法的实例对象的引用即可,看看书中的这段代码:

(function() {
  function _$(els) {
    this.elements = [];
    for (var i = 0, len = els.length; i len; ++i) {
      var element = els[i];
      if (typeof element == 'string') {
        element = document.getElementById(element);
      }
      this.elements.push(element);
}
  };
  _$.prototype = {
    each: function(fn) {
      for ( var i = 0, len = this.elements.length; i len; ++i ) {
        fn.call(this, this.elements[i]);
      }
      return this;
    },
    setStyle: function(prop, val) {
      this.each(function(el) {
        el.style[prop] = val;
      });
      return this;
    },
    show: function() {
      var that = this;
      this.each(function(el) {
        that.setStyle('display', 'block');
      });
      return this;
    },
    addEvent: function(type, fn) {
      var add = function(el) {
        if (window.addEventListener) {
          el.addEventListener(type, fn, false);
        }
        else if (window.attachEvent) {
          el.attachEvent('on'+type, fn);
        }
      };
      this.each(function(el) {
        add(el);
      });
      return this;
    }
  };
  window.$ = function() {
    return new _$(arguments);
  };
})();

可以看到,每个方法都以return this结束,这就会将调用方法的对象传递给链上的下一个方法。但是,如果我们要操作的数据是通过异步请求来获得的,如何保持方法的链式调用呢?Dustin Diaz为我们提供了一种方法来保证方法的链式调用,他也是《javascript设计模式》一书的作者之一。

他首先构建了一个Queue对象,即:


function Queue() {
  // store your callbacks
  this._methods = [];
  // keep a reference to your response
  this._response = null;
  // all queues start off unflushed
  this._flushed = false;
}

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/webkaifa/)

Queue.prototype = {
  // adds callbacks to your queue
  add: function(fn) {
    // if the queue had been flushed, return immediately
    if (this._flushed) {
      fn(this._response);

    // otherwise push it on the queue
    } else {
      this._methods.push(fn);
    }
  },

  flush: function(resp) {
    // note: flush only ever happens once
    if (this._flushed) {
      return;
    }
    // store your response for subsequent calls after flush()
    this._response = resp;
    // mark that it's been flushed
    this._flushed = true;
    // shift 'em out and call 'em back
    while (this._methods[0]) {
      this._methods.shift()(resp);
    }
  }
};

然后用它作为工具构建我们的异步方法队列链。有了这个工具,就可以很方便的构建一个从服务器端获取内容并将其附加到选择器中的jQuery plugin。

(function($) {

  $.fn.fetch = function(url) {
    var queue = new Queue;
    this.each(function() {
      var el = this;
      queue.add(function(resp) {
        $(el).html(resp);
      });
    });
    $.ajax({
      url: url,
      dataType: 'html',
      success: function(html) {
        queue.flush(html);
      }
    });
    return this;
  };

})(jQuery);

这样,我们就可以异步的获取内容,并继续我们的链式调用。

$("div/")
  .fetch('/server/navigation.html')
  .addClass('column')
  .appendTo('#side');

查看demo页看看效果。

如果一个队列中有很多项等待对服务器端的响应进行操作,该如何处置?作者构建了这样一个方法,值得参考:

function fetchTweet(url) {
  this.queue = new Queue;
  this.tweet = "";
  var self = this;
  ajax(url, function(resp) {
    self.tweet = resp;
    self.queue.flush(this);
  });
}
fetchTweet.prototype = {
  linkify: function() {
    this.queue.add(function(self) {
      self.tweet = self.tweet.replace(/b@(w{1,20}b/g, '$1');
    });
  return this;
  },
  filterBadWords: function() {
    this.queue.add(function(self) {
      self.tweet = self.tweet.replace(/b(fuck|shit|piss)b/g, "");
    });
  return this;
  },
  appendTo: function(selector) {
    this.queue.add(function(self) {
      $(self.tweet).appendTo(selector);
    });
  return this;
  }
};

这样,我们就可以用下面的方式来调用:

fetchTweet(url).linkify().filterBadWords().appendTo('#status');
到此,我们已经知道了如何实现异步方法链式调用,但在《Asynchronous method queue chaining in JavaScript》底部的一些评论提出的一些问题,值得思考一下。插件$.fn.fetch中仅仅只需将返回的内容附加到元素之中,Queue是否必要?而且,jQuery中的$.fn.load完全可以实现,如果Queue中只用一个回调函数,完全可以这样来写:

(function($) {
 $.fn.fetch = function(url) {
     var queue = new Queue;
     this.each(function() {
      var el = this;
       $.ajax({
  url: url,
  type: 'get',
  dataType: 'json',
  success: function(resp) {
   $(el).html(resp['text1']);
  }
  });
 });
         return this;
        };
})(jQuery);

不知你作如何感想?

原文:http://www.dustindiaz.com/async-method-queues/
译文:http://www.denisdeng.com/?p=978

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/webkaifa/)

来源:http://www.tulaoshi.com/n/20160220/1633305.html

延伸阅读
标签: Web开发
?php//// +----------------------------------------------------------------------+// | JS javascript 类 |// +----------------------------------------------------------------------+// | Copyright (c) 2001 NetFish Software |// | ...
标签: Web开发
在网上看到有不少JS设计模式的示例。 今天参照: http://www.cnblogs.com/iloveu/archive/2009/03/31/1426234.html 写了一下,记录在此,仅作备忘: !DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd" html xmlns="http://www.w3.org/1999/xhtml...
标签: Web开发
function pCall()  {   __doPostBack('Button1','');  } 可以直接刷新页面,执行后台代码。 (控件名如果含有":" ,需要预先替换为"$") 即使控件设为隐藏,(如button的visible = false),页面上没有生成button, 仍然可以这样直接使用。
标签: Web开发
function getDynamicEnitity(entityName, entityId,attributeName) { var xml = "" + "?xml version="1.0" encoding="utf-8"?" + "soap:Envelope xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema"" + GenerateAuthent...
标签: Delphi
  在用Delphi30开发软件时,出现了硬件驱动程序 (DLL)中的函数和过程不能正常调用的问题,该硬件由英国Schlumberger公司生产,驱动程序用汇编语言编写的。其《编程指南》给出的 Microsoft C的示范程序均能正常运行。但运行此软件时现出的错误提示为: Access violation at address ×××××××× in module…… 经仔细分析...

经验教程

854

收藏

35
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部