其实关于promise 的博客,前端时间专门写了一篇关于 promise 规范的文章,promise规范 让 javascript 中的异步调用更加人性化.

简单回忆下:

promise/A规范定义的“Deferred/Promise”模型

其实是实现了“发布/订阅”模型.

通过Deferred对象发布事件,

包括下面2种事件

完成 --> resolve事件,

失败 --> reject事件,

通过Promise对象进行对应完成或失败的订阅

类似于任务状态转变时事件被触发.

将会对应的执行事件绑定函数.

每个Promise任务都有三种状态:

默认(pending)、完成(fulfilled)、失败(rejected)

1.默认状态可以单向转移到完成状态,这个过程叫resolve,对应的方法是deferred.resolve(Value);

2.默认状态还可以单向转移到失败状态,这个过程叫reject,对应的方法是deferred.reject(reason);

3.默认状态时,还可以通过deferred.notify(message)来宣告任务执行信息,如执行进度;

4.状态的转移是一次性的,一旦任务由初始的pending转为其他状态,就会进入到下一个任务的执行过程中。

今天其实是想和大家分享下 when 这个 npm 包,或许未来不久你将会用到.

github地址: https://github.com/cujojs/when

我们通过一个具体的模拟示例来学习 when

应用场景:

假如我有一个服务,每天定时运行.

做什么任务呢,就是去证券/银行站点读取当天的汇率信息,然后保存到自己的数据库.

逻辑很简单:

  1. 查看当天是否有最新汇率信息

  2. 如果有,获取汇率信息

  3. 如果获取到,保存到本地数据库

我们依次来定义3个方法:

//获取汇率信息
function get(callback){
    //具体请求url,分析dom全部省略,全当调用了下面的一个黑盒方法来实现
    yijiebuyi_util.getHtml('http://aaa.com/bbb',function(err,info){
        callback(err,info);
    });
}
//保存汇率信息
function save(info,callback){
    //写入数据库方法忽略
    yijiebuyi_util.save(info,function(err,result){
        callback(err,result);
    });
}

OK,我们先来一把普通调用方法

function main(callback){
    //获取汇率信息
    get(function(err,info){
       if(err){
           return callback(err,null);
       } 
       //保存汇率信息
       save(info,function(err,result){
           if(err){
               return callback(err,null);
           }
           //回调保存状态
           callback(null,result);
       });
    });
};

如果业务更复杂的话,一层嵌套一层,看上去是不很带劲等到维护的时候可能就费劲了.

有人说要使用 async 来控制,确实可行,而且我们一般也就是这样处理的.

async 可以把整个流程控制住,比如串联执行,下一步执行函数使用上一步执行结果等.

但是今天我们推荐大家一种看上去很优雅的解决方案.

我们从处理流程跳出来,不要在流程上去控制业务,而是从每一个业务方法入手,让方法通过 when 构建一个promise规范的方法.

我们还是要把上面的方法全部修改一下:

首先安装,应用 when

npm install -g when
var when=require('when');
function get(){
    var deferred = when.defer();
    yijiebuyi_util.getHtml('http://aaa.com/bbb',function(err,info){
        if(err)
            deferred.reject(err);
        else
            deferred.resolve(info);
            
        return deferred.promise;
    });
}
function save(info,callback){
    var deferred = when.defer();
    yijiebuyi_util.save(info,function(err,result){
        if(err)
            deferred.reject(err);
        else
            deferred.resolve(result);
            
        return deferred.promise;
    });
}

上面2个方法我们已经改装完毕,下面看看 main 方法中如何调用呢.

function main(){
    get().then(save).catch(function(err){
        console.log(err);
    });
}

我们并没有在流程上控制业务逻辑处理顺序.而是像同步代码一样按照执行顺序调用即可,是不是很美好.

then方法包括三个参数,

onFulfilled、

onRejected、

onProgress

promise.then(onFulfilled, onRejected, onProgress)

从参数名你就可以发现这几个方法应该是事件处理函数.

这也应了我们上面所说的, promise 本身就是一种事件发布/订阅模型.

所以上面 then 里的三个函数相当于事件绑定函数.(就是观察者)

当上一个任务被deferred.resolve(data) ,对应在本任务就会触发 onFulfilled 方法.

当上一个任务被 deferred.reject(err) ,对应的本任务就会触发 onRejected 方法.

任何一个任务, onFulfilled 或 onRejected 方法只能被触发一个,并且触发一次.

我们上面main 方法中最后的 catch(function(err){.....}) 是怎么回事.

when 提供了极其简单的传递错误机制,多个任务执行时,我们可以在最后一个任务定义onRejected

或者在 then 调用的最后用 catch 来捕捉错误.

***end***

when 让你跳出异步回调噩梦 node.js下promise/A规范的使用的更多相关文章

  1. 深入浅出Node.js(下)

    (五):Node.js的异步实现 专栏的第五篇文章<Node.js的异步实现>.之前介绍了Node.js的事件机制,也许读者对此尚会觉得意犹未尽,因为仅仅只是简单的事件机制,并不能道尽No ...

  2. node.js 下依赖Express 实现post 4种方式提交参数

    上面这个图好有意思啊,哈哈, v8威武啊.... 在2014年的最后一天和大家分享关于node.js 如何提交4种格式的post数据. 上上一篇说到了关于http协议里定义的4种常见数据的post方法 ...

  3. npm 是node.js下带的一个包管理工具

    npm 是node.js下带的一个包管理工具          npm install -g webpack webpack是一个打包工具 gulp是一个基于流的构建工具,相对其他构件工具来说,更简洁 ...

  4. node.js下操作cookie

    cookie,又是cookie.工作中与cookie打交道很多次,不过时间跨度也大,每总结多一次,就加深了解多一点. cookie,一定是放在浏览器中的,用于浏览器保存一些小额度的内容.每次我们去访问 ...

  5. Node.js之Promise维护(同步)多个回调(异步)状态

    金天:学习一个新东西,就要持有拥抱的心态,如果固守在自己先前的概念体系,就会有举步维艰的感觉..NET程序员初用node.js最需要适应的就是异步开发, 全是异步,常规逻辑下遍历列表都是异步,如何保证 ...

  6. 避免多层回调,Node.js异步库Async使用(series)

    未使用Async之前coffeescript写的代码: exports.product_file_add = (req,res) -> if !req.param('file_id') retu ...

  7. node.js的Promise库-bluebird示例

    前两天公司一哥们写了一段node.js代码发给我,后面特意提了一句“写的不太优雅”.我知道,他意思是回调嵌套回调,因为当时比较急也就没有再纠结.然而内心中总记得要解决这个问题.解决node.js的回调 ...

  8. Node.js下的Hello World

    Node.js技术现在可谓是如火如荼,前后端都统一为Javascript的体验绝对是受到了很多人的青睐,我都后悔以前没抽时间好好学一学Javascript了. 首先,我来介绍一下Node.js.本人实 ...

  9. 在Node.js使用Promise的方式操作Mysql

    最近在学习Node.js,虽然早就听说了回调地狱结果过了一周就遇到了.所以花时间学习了了一下Promise.虽然还有Async/await.co.生成器等选择,但是因为本人基础较差,以及时间问题所以决 ...

随机推荐

  1. 惠普台式机在UEFI BIOS设置通电自动开机 影响电脑自动重启关不了机设置

    设置通电自动开机 影响电脑自动重启关不了机设置   惠普台式机在UEFI BIOS中 1. 开机时不断点击F10键进入BIOS,选择Advanced(高级)然后选择Boot Options,点击回车 ...

  2. Day03 知识点

    一.单行注释与多行注释 # 用来标记不运行的程序 pycharm 快捷键 ctrl+/ 可以在程序上方 也可以在程序后面 (PEP8) 多行注释 用三引号,一般推荐三双引号来做注释. 二.数据类型 ( ...

  3. [原创]hibernate更新后jdbc读取不到数据问题

    最近在做工作流插件时使用的是自己基于hibernate连接封装的orm框架,按说跟hibernate共用的一个连接,应该在同一个事务中,但在使用时hibernate saveOrUpdate后(未提交 ...

  4. Kudu的架构

    不多说,直接上干货!  Kudu的架构 1.kudu的 基本框架 Kudu 是用于存储结构化( structured )的表( Table ).表有预定义的带类型的列( Columns ),每张表有一 ...

  5. 纯CSS控制背景图片100%自适应填充布局

    https://blog.csdn.net/wd4java/article/details/50537562 解决:   html,body{height: 100%;width: 100%;marg ...

  6. 如何快速定位JVM中消耗CPU最多的线程?

    第一步.先找出Java的进程PID ps -ef | grep 进程名关键字 这里假设找到的PID是:12345   第二步.找出该进程内最消耗CPU的线程 top -Hp log4x R :11.7 ...

  7. Linux如何上查看和退出tomcat实时日志

    1.先切换到:cd usr/local/tomcat/logs 目录下 2.tail -f catalina.out Ctrl+c 是退出tail命令. alt+E+R重置.

  8. centos系统安装mysql

    方式一. 通过yum install mysql-server安装mysql服务器.chkconfig mysqld on设置开机启动,并service mysqld start启动mysql服务,并 ...

  9. WebSocket Demo

    HTML 代码: <body> <h1>WebScoket示例</h1> <br /><br /> <input type=" ...

  10. vue-2.4.0-添加的新东东

    组件内新增实现属性继承 VUE中一个比较令人烦恼的事情是属性只能从父组件传递给子组件.这也就意味着当你想向嵌套层级比较深组件数据传递,只能由父组件传递给子组件,子组件再传递给孙子组件...像下面这样 ...