异步流程控制库GoWithTheFlow
异步流程控制库GoWithTheFlow
一个尾触发方式来控制异步流程的库, 有seq(顺序执行) par(同步执行) 两种方法
博客
http://notes.jetienne.com/2011/07/17/gowiththeflow.js-async-flow-control-with-a-zen-touch.html
Github
https://github.com/lumixraku/gowiththeflow.js/blob/master/gowiththeflow.js
使用
顺序执行
Flow().seq(function(next) {
setTimeout(function(){
console.log("first job");
next('error', 'retValue');
},500);
}).seq(function(next, error, result) {
console.log(error, result);
console.log("second job. run *after* first job");
next();
});
同步执行
最后挂载的seq将会在所有par任务执行结束后执行
errors results是数组 保存所有的par的结果
执行结果顺序是挂载任务的顺序
Flow().par(function(next){
setTimeout(function(){
console.log("job foo");
next(null, "foo");
},1000);
}).par(function(next){
setTimeout(function(){
console.log("job bar");
next(null, "bar");
},500);
}).par(function(next){
setTimeout(function(){
console.log("job zoo");
next(null, "zoo");
},500);
}).seq(function(next, errors, results){
console.log("job run *after* the completion of foo and bar");
console.assert(errors.length == 3 && errors[0] === null && errors[1] == null)
console.assert(results.length == 3 && results[0] === 'foo' && results[1] == 'bar')
next();
})
实现
var Flow = function() {
var self, stack = [],
//等待所有同步操作完成后开始执行异步的部分
timerId = setTimeout(function() {
timerId = null;
self._next();
}, 0);
return self = {
destroy: function() {
timerId && clearTimeout(timerId);
},
//seq 和par 都只是挂载了函数(把函数保存在了stacks中) 并没有执行
//同步执行的任务都在一个元素中
//比如 stack = [ [f1()], [f2(), f3()], [f4()] ]
//表示 f1() f2()f3() f4() 三个顺序执行的任务 其中任务2 f2f3两个函数是并行执行
par: function(callback, isSeq) {
if (isSeq || !(stack[stack.length - 1] instanceof Array)) {
stack.push([]);
}
stack[stack.length - 1].push(callback);
return self; //链式调用
},
seq: function(callback) {
return self.par(callback, true);
},
//调用一次_next 解决一次顺序任务
_next: function(err, result) {
var errors = [],
results = [],
callbacks = stack.shift() || [], //callbacks [function(next){....}]
nbReturn = callbacks.length,
isSeq = nbReturn == 1; //表明是顺序任务 比如上面 [f1()] 这种情况
for (var i = 0; i < callbacks.length; i++) {
(function(fct, index) {
//fct就是异步函数 fct接受三个参数 fct(next, err, result)
//而这个function(err, result) 就是next
//next接受两个参数 next('error', 'retValue');
fct(function(error, result) {
errors[index] = error;
results[index] = result;
if (--nbReturn == 0) { //表明此次顺序任务执行完毕 比如上面[f2,f3] 到f3的时候
self._next(isSeq ? errors[0] : errors, isSeq ? results[0] : results)
}
}, err, result);
})(callbacks[i], i);
}
}
}
};
// export in common js
if (typeof module !== "undefined" && ('exports' in module)) {
module.exports = Flow;
}
// Asynchronous Module Definition - http://requirejs.org/docs/whyamd.html
if (typeof define === 'function' && define.amd) {
define('Flow', [], function() {
return Flow;
});
}
异步流程控制库GoWithTheFlow的更多相关文章
- js 异步流程控制之 avQ(avril.queue)
废话前言 写了多年的js,遇到过最蛋疼的事情莫过于callback hell, 相信大家也感同身受. 业界许多大大也为此提出了很多不错的解决方案,我所了解的主要有: 朴灵 event proxy, 简 ...
- node核心:异步流程控制
Node.js的异步是整个学习Node.js过程中重中之重. 1)异步流程控制学习重点 2)Api写法:Error-first Callback 和 EventEmitter 3)中流砥柱:Promi ...
- nodejs进阶(7)—async异步流程控制
Async介绍 Async是一个流程控制工具包,提供了直接而强大的异步功能.基于Javascript为Node.js设计,同时也可以直接在浏览器中使用. Async提供了大约20个函数,包括常用的 m ...
- 使用yield进行异步流程控制
现状 目前我们对异步回调的解决方案有这么几种:回调,deferred/promise和事件触发.回调的方式自不必说,需要硬编码调用,而且有可能会出现复杂的嵌套关系,造成"回调黑洞" ...
- 【javascript】Promise/A+ 规范简单实现 异步流程控制思想
——基于es6:Promise/A+ 规范简单实现 异步流程控制思想 前言: nodejs强大的异步处理能力使得它在服务器端大放异彩,基于它的应用不断的增加,但是异步随之带来的嵌套.难以理解的代码让 ...
- Nodejs中使用异步流程控制Async
首先,我们都知道,Node基于事件驱动的异步I/O架构,所谓异步就是非阻塞,说白了就是一个事件执行了,我不必等待它执行完成后我才能执行下一个事件.所以在Node环境中的模块基本都是异步的,上一篇说到我 ...
- node基础13:异步流程控制
1.流程控制 因为在node中大部分的api都是异步的,比如说读取文件,如果采用回调函数的形式,很容易造成地狱回调,代码非常不容易进行维护. 因此,为了解决这个问题,有大神写了async这个中间件.极 ...
- (一)Nodejs - 框架类库 - Nodejs异步流程控制Async
简介 Async是一个流程控制工具包,提供了直接而强大的异步功能 应用场景 业务流程逻辑复杂,适应异步编程,减少回调的嵌套 安装 npm insatll async 函数介绍 Collections ...
- async 异步流程控制规则
github 学习async网址 : https://github.com/caolan/async/ 1.Async 函数介绍 async 主要实现了三个部分的流程控制功能 1.集合:Collect ...
随机推荐
- GitBook是一个命令行工具(Node.js库),我们可以借用该工具使用Github/Git和Markdown来制作精美的图书,但它并不是一本关于Git的教程哟。
GitBook是一个命令行工具(Node.js库),我们可以借用该工具使用Github/Git和Markdown来制作精美的图书,但它并不是一本关于Git的教程哟. 支持输出多种格式 GitBook支 ...
- js创建对象的方式 三种
1. 使用直接量创建1个对象: var aobj = { x : 10, y : function(){ console.log("aobj--> "+this.x); } ...
- thbgm拆包【in progress】
曾经在网上找过但是没找到过....关于东方系列bgm的格式,最初以为是个加密格式,后来听说是多个wav堆到一块儿的.再后来查到有说可以用GoldWave开的.今天试了试成功了.接下来打算研究一下,不过 ...
- Server2008系统 FTP下载“当前的安全设置不允许”的解决方法
IE -> Internet选项 -> 安全 -> Internte -> 自定义级别 设置 -> 下载 -> 文件下载 -> 启动
- 基于jQuery的前端如何做到无伤迁移
首先,解释一下我个人对前端无伤迁移的理解,即移动端和PC端使用同一套代码,或者说原本在PC端运行得很完美的代码,只要修改少许,就可以在移动端完美运行. 当然,大部分的公司会专门为移动端设计了一套,同时 ...
- Android周笔记(9.8-14)(持续更新)
本笔记记录一周内的小知识点和一些心学习的Demo. 1.PopupWindow: new 一个activity_pop_window:id为popwindow的Button,id为hello123的T ...
- Django模板-分离的模板
上一篇Django模板-在视图中使用模板最后的问题,我们需要把数据和展现分离开. 你可能首先考虑把模板保存在文件系统的某个位置并用 Python 内建的文件操作函数来读取文件内容. 假设文件保存在 E ...
- jquery.post方法回调函数
1 function(data){} 此post请求成功后调用之,data是请求成功后服务器返回的东西.如果在servlet中有response.getWriter().println("s ...
- 浅析C++内存分配与释放操作过程——三种方式可以分配内存new operator, operator new,placement new
引言:C++中总共有三种方式可以分配内存,new operator, operator new,placement new. 一,new operator 这就是我们最常使用的 new 操作符.查看汇 ...
- CSS预处理器实践之Sass、Less比较
什么是CSS预处理器? CSS可以让你做很多事情,但它毕竟是给浏览器认的东西,对开发者来说,Css缺乏很多特性,例如变量.常量以及一些编程语法,代码难易组织和维护.这时Css预处理器就应运而生了.Cs ...