原promise.js库地址:https://github.com/stackp/promisejs

promises是JavaScript实现优雅编程的一个非常不错的轻量级框架。该框架可以让你从杂乱的多重异步回调代码中解脱出来,并把精力集中到你的业务逻辑上。

今天从GIT源码库中下载了promise.js,发现该源码是基于Web前端JavaScript写的,并不能直接用于nodejs。还好代码不是很多,也不是很复杂。经过分析整合,将其实现为nodejs的一个框架,代码如下:

(function(){
/**
* Copyright 2012-2013 (c) Pierre Duquesne <stackp@online.fr>
* script: promise.js
* description: promises的nodejs模块
* modified: https://github.com/stackp/promisejs
* authors: alwu007@sina.cn
* */

var Promise = exports.Promise = function(){
    this._callbacks = [];
};

Promise.prototype.then = function(func, context){
    //处理回调结果的方法
    function doCallbackResults(r) {
        if (r instanceof Promise) {
            r.then(function(err, values){
                p.done(err, values);
            });
        } else {
            p.done(null, r);
        }
    }

var p = new Promise();
    if (this._isdone) {
        var results = func.apply(context, this.results);
        doCallbackResults(results);
    } else {
        this._callbacks.push(function(){
            var results = func.apply(context, arguments);
            doCallbackResults(results);
        });
    }
    return p;
};

Promise.prototype.done = function(){
    this.results = arguments;
    this._isdone = true;
    for (var i=0; i<this._callbacks.length; i++) {
        this._callbacks[i].apply(null, arguments);
    }
    this._callbacks = [];
};

Promise.join = function(promises){
    var p = new Promise();
    var results = [];

if (!promises || !promises.length) {
        p.done(results);
        return p;
    }

var numdone = 0;
    var total = promises.length;

function notifier(i) {
        return function() {
            numdone += 1;
            results[i] = Array.prototype.slice.call(arguments);
            if (numdone === total) {
                p.done(results);
            }
        };
    }

for (var i = 0; i < total; i++) {
        promises[i].then(notifier(i));
    }

return p;
};

Promise.chain = function(funcs, args) {
    var p = new Promise();
    if (!funcs || !funcs.length) {
        p.done.apply(p, args);
    } else {
        funcs[0].apply(null, args).then(function(){
            funcs.splice(0, 1);
            Promise.chain(funcs, arguments).then(function(){
                p.done.apply(p, arguments);
            });
        });
    }
    return p;
};
})();

另附测试代码如下:

/**
* script: test.js
* description: promise.js测试代码
* */

var promise = require('./mypromise');

function asyncfoo() {
    var p = new promise.Promise();
    setTimeout(function(){
        p.done();
    }, 1000);
    return p;
}

function syncfoo() {
    var p = new promise.Promise();
    p.done();
    return p;
}

var o = {};
/*
asyncfoo().then(function(){
    return 'Raymond';
}, o).then(function(err, name){
    o.name = name;
    return asyncfoo().then(asyncfoo).then(function(){
        return asyncfoo().then(asyncfoo).then(function(){
            return 18;
        });
    });
}, o).then(function(err, age){
    o.age = age;
    return asyncfoo().then(asyncfoo).then(function(){
        return asyncfoo().then(asyncfoo).then(function(){
            return 'boy';
        });
    }).then(function(err, sex){
        return sex;
    });
}).then(function(err, sex){
    o.sex = sex;
    return 'Hello, world!';
}).then(function(err, say){
    o.say = say;
    console.dir(o);
});

syncfoo().then(function(){
    return 'Raymond';
}, o).then(function(err, name){
    o.name = name;
    return syncfoo().then(syncfoo).then(function(){
        return syncfoo().then(syncfoo).then(function(){
            return 18;
        });
    });
}, o).then(function(err, age){
    o.age = age;
    return asyncfoo().then(asyncfoo).then(function(){
        return asyncfoo().then(asyncfoo).then(function(){
            return 'boy';
        });
    }).then(function(err, sex){
        return sex;
    });
}).then(function(err, sex){
    o.sex = sex;
    return 'Hello, world!';
}).then(function(err, say){
    o.say = say;
    console.dir(o);
});
*/
function asyncfoo1(){
    var p = new promise.Promise();
    setTimeout(function(){
        p.done(null, 'Raymond');
    }, 1000);
    return p;
}

function asyncfoo2(err, name){
    o.name = name;
    var p = new promise.Promise();
    setTimeout(function(){
        p.done(null, 18);
    }, 1000);
    return p;
}
function asyncfoo3(err, age){
    o.age = age;
    var p = new promise.Promise();
    setTimeout(function(){
        p.done(null, 'boy');
    }, 1000);
    return p;
}
function asyncfoo4(){
    var p = new promise.Promise();
    setTimeout(function(){
        p.done(null, 'Hello, world!');
    }, 1000);
    return p;
}
promise.Promise.chain([asyncfoo1, asyncfoo2, asyncfoo3]).then(function(err, sex){
    o.sex = sex;
    return asyncfoo4();
}).then(function(err, say){
    o.say = say;
}).then(function(){
    console.dir(o);
});

最新文章

  1. 用信息值进行特征选择(Information Value)
  2. Web Service 的创建简单编码、发布和部署
  3. mysql环境搭建
  4. phpmyadmin 长时间登陆不过期
  5. oracle 11gr2 官方文档下载
  6. 05.K米评测
  7. Autofac IContainer 测试
  8. 8个主要的Velocity语法使用说明
  9. 收集Magento FAQ常见问题处理办法
  10. (Excel导出失败)检索COM类工厂中CLSID为{00024500-0000-0000-C000-000000000046}的组件时失
  11. 最近面了不少java开发,据此来说下我的感受:哪怕事先只准备1小时,成功概率也能大大提升
  12. 关于flask 上直接使用py.test测试框架进行测试
  13. 2 c++对象被使用前要先被初始化
  14. [Spark Streaming_1] Spark Streaming 概述
  15. 强大的安卓手机远程管理工具 – Droidjack
  16. maven scope &#39;provided&#39; 和 ‘compile’的区别
  17. C++11学习
  18. 移动对meta的定义(转)
  19. Outlook中在Exchange服务器无法保存邮件副本
  20. IE9 下面, XMLHttpRequest 是不支持跨域请求的解决方法

热门文章

  1. call 方法在使用一个指定的this值和若干个指定的参数值的前提下调用某个函数或方法.
  2. 03:计算(a+b)/c的值
  3. while if 循环判断
  4. js的引用顺序
  5. javabean 简介
  6. python中xrange与range的异同
  7. lpad rpad
  8. 信号槽的被连接几次,就会执行几次(有空要仔细研究connect的各种用法)
  9. Samples DataBind FastJson循环引用问题
  10. 【HDOJ】4704 Sum