设为首页 收藏本站
查看: 1005|回复: 0

[经验分享] NodeJs回调操作Promise化

[复制链接]

尚未签到

发表于 2017-2-24 10:18:47 | 显示全部楼层 |阅读模式
  mongoose是一个NodeJs下MongoDB的ORM库。使用这个库,您从DB到表(collection)都不用创建了。只需要在项目中定义好Model。
  下面就是用上一篇的代码来演示如何把mongoose的数据库操作里的回调地狱(callback hell)轻松化解掉。
  上一篇Petshop的代码在这里。

打开Promise的开关
  mongoose已经开启了对Promise的支持,只需要指定明确的Promise库就可以:
var mongoose = require('mongoose'),  Promise = require('bluebird');
  本来每一个model的定义都需要引入mongoose库,然后每次都给mongoose库指定Promise库太过冗繁。所以我们抽象代码,在models目录下创建一个base目录,然后在里面添加一个index.js文件:
//petshop/server/models/index.js  

  
var mongoose = require('mongoose'),
  Promise = require('bluebird');
  
mongoose.Promise = Promise;
  

  
module.exports = mongoose;
  然后在model的定义都是用export的添加Promise的mongoose:
var mongoose    = require('./base'),  bcrypt      = require('bcrypt-nodejs');
  

  
var Schema = mongoose.Schema;
  

  
var userSchema = new Schema({
  username: {type: String, unique: true, required: true},
  password: {type: String, required: true}
  
});
  

  
...
  

  
module.exports = mongoose.model('User', userSchema);
  这样,使用了base目录下的mongoose定义的model都具备了Promise的能力。
  在调用查找更新等方法的时候只需要这样:
    User.findOne({ username: username }).exec().then(function (u) {  if (!u) {
  done(null, false);
  return;
  }
  var verifyPasswordAsync = Promise.promisify(u.verifyPassword, { context: u });
  verifyPasswordAsync(password).then(function (match) {
  console.log('password match ' + match);
  if (!match) {
  console.log('is match ' + match);
  done(null, false);
  } else {
  done(null, u);
  }
  });
  }).catch(function (err) {
  done(err);
  });
  解释如下:
  
第一行代码User.findOne({ username: username }).exec()在exec调用之后就返回了一个Promise。后面就可以使用Promise的then方法来开始Promise的方式依次调用和异常处理了。

单独promise化一个方法
  在mongoose内置的Promise支持不能完成某些方法的时候还可以另外使用bluebird库来单独的针对这个方法来使其promise化。比如上例的u.verifyPassword代码:
userSchema.methods.verifyPassword = function (password, callback) {  bcrypt.compare(password, this.password, function (err, match) {
  if (err) {
  return callback(err);
  }
  

  callback(null, match);
  });
  
};
  单独的promise化verifyPassword方法:
var verifyPasswordAsync = Promise.promisify(u.verifyPassword, { context: u });  之后的使用:
verifyPasswordAsync(password).then(function (match) {  console.log('password match ' + match);
  if (!match) {
  console.log('is match ' + match);
  done(null, false);
  } else {
  done(null, u);
  }
  
});

Promise化的一般原则
  对于上面例子这里稍作引申。Promise化的时候使用的是bluebird库。
  下面使用的例子代码如下:
function Dog(name) {  this.name = !name ? 'Tiger': name;
  
}
  

  
Dog.prototype.bite = function(target, cb){
  console.log(this.name + ' bite ' + target);
  cb(null, target);
  
};

Promise化一个对象
  Promise化一个对象使用promisifyAll方法。
var Promise = require('bluebird');  

  
var d = Promise.promisifyAll(new Dog());
  
d.biteAsync('hellokitty');
  输出:
  

Tiger bite hellokitty  

  注意:Promise化之后调用方法需要加上Async后缀。bite=>biteAsync。

Promise化一个方法
  Promise化的是一个带有回调的方法。这个Promise返回的结果就是回调正确的情况下获得的值。
var someDog = new Dog("small");  
var otherDog = new Dog("big");
  
var proDog = Promise.promisify(someDog.bite, {context: otherDog});
  
proDog('YOU').then(function(target) {
  console.log('then ' + target);
  walk();
  
})
  在Promise话一个方法的时候需要考虑是不是指定context。上例中如果不指定context的时候会报错。一般,如果是require引入的库的方法不需要指定context,但是局部变量需要制定。指定context以后,方法的this指向的上下文就是这个context对象。

总结
  Promise化之后,回调地狱的问题就有很好的解决了。不过,还需要考虑项目的大小和回调的深度来决定是否要Promise化。毕竟Promise会增加一定的代码量,也有一定的学习曲线。
  

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-346507-1-1.html 上篇帖子: Nodejs中Mongodb的基本使用 下篇帖子: Nodejs进阶:如何玩转子进程(child_process)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表