异步流程控制库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的更多相关文章

  1. js 异步流程控制之 avQ(avril.queue)

    废话前言 写了多年的js,遇到过最蛋疼的事情莫过于callback hell, 相信大家也感同身受. 业界许多大大也为此提出了很多不错的解决方案,我所了解的主要有: 朴灵 event proxy, 简 ...

  2. node核心:异步流程控制

    Node.js的异步是整个学习Node.js过程中重中之重. 1)异步流程控制学习重点 2)Api写法:Error-first Callback 和 EventEmitter 3)中流砥柱:Promi ...

  3. nodejs进阶(7)—async异步流程控制

    Async介绍 Async是一个流程控制工具包,提供了直接而强大的异步功能.基于Javascript为Node.js设计,同时也可以直接在浏览器中使用. Async提供了大约20个函数,包括常用的 m ...

  4. 使用yield进行异步流程控制

    现状 目前我们对异步回调的解决方案有这么几种:回调,deferred/promise和事件触发.回调的方式自不必说,需要硬编码调用,而且有可能会出现复杂的嵌套关系,造成"回调黑洞" ...

  5. 【javascript】Promise/A+ 规范简单实现 异步流程控制思想

    ——基于es6:Promise/A+ 规范简单实现 异步流程控制思想  前言: nodejs强大的异步处理能力使得它在服务器端大放异彩,基于它的应用不断的增加,但是异步随之带来的嵌套.难以理解的代码让 ...

  6. Nodejs中使用异步流程控制Async

    首先,我们都知道,Node基于事件驱动的异步I/O架构,所谓异步就是非阻塞,说白了就是一个事件执行了,我不必等待它执行完成后我才能执行下一个事件.所以在Node环境中的模块基本都是异步的,上一篇说到我 ...

  7. node基础13:异步流程控制

    1.流程控制 因为在node中大部分的api都是异步的,比如说读取文件,如果采用回调函数的形式,很容易造成地狱回调,代码非常不容易进行维护. 因此,为了解决这个问题,有大神写了async这个中间件.极 ...

  8. (一)Nodejs - 框架类库 - Nodejs异步流程控制Async

    简介 Async是一个流程控制工具包,提供了直接而强大的异步功能 应用场景 业务流程逻辑复杂,适应异步编程,减少回调的嵌套 安装 npm insatll async 函数介绍 Collections ...

  9. async 异步流程控制规则

    github 学习async网址 : https://github.com/caolan/async/ 1.Async 函数介绍 async 主要实现了三个部分的流程控制功能 1.集合:Collect ...

随机推荐

  1. linux常用命令详解 (二)文件处理命令

    ◆ 文件处理命令:file.mkdir.grep.dd.find.mv.ls.diff.cat.ln: 系统信息存放在文件里,文件与普通的公务文件类似.每个文件都有自己的名字.内容.存放地址及其它一些 ...

  2. java总结

    JUC概况 以下是Java JUC包的主体结构: ? Atomic : AtomicInteger ? Locks : Lock, Condition, ReadWriteLock ? Collect ...

  3. HDU 1045(Fire Net)题解

    以防万一,题目原文和链接均附在文末.那么先是题目分析: [一句话题意] 给定大小的棋盘中部分格子存在可以阻止互相攻击的墙,问棋盘中可以放置最多多少个可以横纵攻击炮塔. [题目分析] 这题本来在搜索专题 ...

  4. 阿牛的EOF牛肉串

    #include <iostream>using namespace std;long long s0,s1,s2,s3;int main(){ int i,n; while(cin> ...

  5. Ubuntu package managerment tools

    Visual demostration References Understanding differences between dpkg and apt-get/aptitude tools. A ...

  6. Selenium2Library使用Remote功能(转载并更新)

    在selenium2library库的open browser中,除了我们常用的url,browser外,还有几个不常用的参数.如:remote_url的用法 1.下载selenium-server- ...

  7. 论山寨手机与Android 【12】3G时代SmartPhone BP部分

    最成熟的3G网络系统,是3GPP项目组制订的WCDMA.WCDMA的网络结构,可参考Figure 12.1,其中有几个特点. 1. 反向兼容GSM/GRPS网络. 原有GSM网络的基站子系统(BSS) ...

  8. Object.prototype.toString.call() 区分对象类型(判断对象类型)

    在 JavaScript 里使用 typeof 来判断数据类型,只能区分基本类型,即 “number”,”string”,”undefined”,”boolean”,”object” 五种.对于数组. ...

  9. 在OpenCV中利用鼠标绘制矩形和截取图像的矩形区域

    这是两个相关的程序,前者是后者的基础.实际上前一个程序也是在前面博文的基础上做的修改,请参考<在OpenCV中利用鼠标绘制直线> .下面贴出代码. 程序之一,在OpenCV中利用鼠标绘制矩 ...

  10. Backward Digit Sums(暴力)

    Backward Digit Sums Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 5664   Accepted: 32 ...