JavaScript 异步方法队列链实现代码分析

时间:2024-04-22 13:26:14 

在《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;
}
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);


不知你作如何感想?

标签:异步方法,队列链
0
投稿

猜你喜欢

  • Pytorch实现图像识别之数字识别(附详细注释)

    2022-04-08 08:12:54
  • pytorch中的inference使用实例

    2023-07-24 16:22:56
  • redis查看连接数及php模拟并发创建redis连接的方法

    2023-11-16 11:47:14
  • 轻松掌握 SQL Server 2000数据库的构架

    2009-02-05 15:50:00
  • PyTorch中可视化工具的使用

    2021-03-05 23:11:39
  • Python编程技巧连接列表的八种操作方法

    2022-02-10 02:08:54
  • pandas中的series数据类型详解

    2023-11-30 20:30:30
  • Python实现JSON反序列化类对象的示例

    2023-09-03 19:29:51
  • 基于h5py的使用及数据封装代码

    2021-10-22 08:45:39
  • Go语言中转换JSON数据简单例子

    2024-04-26 17:32:28
  • Python检测PE所启用保护方式详解

    2022-03-11 12:36:08
  • 深入分析SQL Server 存储过程

    2024-01-19 21:18:09
  • Mysql中实现提取字符串中的数字的自定义函数分享

    2024-01-25 10:56:20
  • python使用redis模块来跟redis实现交互

    2023-05-06 04:13:40
  • JavaScript使用HTML5的window.postMessage实现跨域通信例子

    2024-04-10 11:00:41
  • 关于Python 多重继承时metaclass conflict问题解决与原理探究

    2022-04-17 04:50:10
  • 10分钟搭建自己的Git仓库

    2023-05-18 21:10:19
  • MySQL表设计优化与索引 (九)

    2010-10-25 20:16:00
  • 浅谈MySQL之select优化方案

    2024-01-27 03:32:38
  • 数据库查询的分页优化技巧

    2009-05-17 10:31:00
  • asp之家 网络编程 m.aspxhome.com