ES6 Promise 很棒。到目前为止,从回调习语调整我的想法很容易。我发现它自然会鼓励更多的模块化代码,当然错误处理也更加清晰。
但是有几次我遇到了似乎(?)无法从 nodebacks 轻松转换为 promises 的流程情况(也许就是这样,但也许我只是对答案视而不见)。由于 Promises 对下一个操作(或者即使有一个操作)是不可知的,所以将 Promises 与不仅接受回调而且还返回它们的 API 一起使用似乎非常困难。
想到的最常见的例子是“完成”回调。它出现在诸如数据库连接之类的东西中,表示“将连接返回到池”,但我也看到它在许多其他地方也出现了。
function getSomeStupidConnection(cb) {
var conn = /* ... */;
var iNeedToBeToldWhenIAmDone = function() { /* ... */ };
cb(conn, iNeedToBeToldWhenIAmDone);
}
getSomeStupidConnection(function(conn, done) {
/* ... */
conn.doLotsOfStuff(function(soMuchStuff) {
/* stuff! so much fun! */
/* okay conn go away I’m tired */
done();
});
});
像这样的流反转显然不是您希望在 API 中开始使用的东西,但它就在那里,有时您无法真正避免它。使用回调,您可以将“稍后调用”内部回调传递给原始“外部”回调。它并不能完全分离关注点,但至少它是快速和简单的。
是否有适合这种情况的基于 Promise 的方法?一种说法,'这是解决值 - 但是当链完成时,也这样做'?我怀疑没有什么与我刚才描述的完全匹配,因为实际上不可能说一个链条已经“完成”,但也许我错过了一些让你接近它而不造成混乱的模式......
编辑:根据到目前为止的反馈,我意识到根本没有办法将这样的 API 包装在真正的 Promise 中,因为您返回的 Promise 将永远无法告诉您任何有关任何后续链式 Promise 的信息。但你可以伪造它。扭曲的是结果相当脆弱。它必须假定唯一then
需要连接对象的是紧随其后的那个。承诺的消费者需要了解它是一次性使用的连接,否则这并不明显。因此,我在实践中并不真正推荐它,但为了好奇,这里有一个解决方案,它隐藏了done
作为(并最终成为)承诺链的同时:
/* jshint node: true, esnext: true */
'use strict';
// Assume this comes from an external library. It returns a connection and a
// callback to signal that you are finished with the connection.
function getConnectionExternal(cb) {
let connection = 'Received connection.';
let done = () => console.log('Done was called.');
cb(null, connection, done);
}
// Our promisey wrapper for the above
function getConnection() {
let _done;
let promise = new Promise((resolve, reject) => {
getConnectionExternal((err, connection, done) => {
if (err) return reject(err);
_done = (val) => {
done();
return val;
};
resolve(connection);
});
});
let _then = promise.then.bind(promise);
promise.then = (handler) => _then(handler).then(_done, _done);
return promise;
}
// Test it out
getConnection()
.then(connection => {
console.log(connection);
return new Promise((resolve, reject) => {
setTimeout(() => {
console.log('Finished using connection!');
resolve('This should be after connection closes.');
}, 200);
});
})
.then(msg => console.log(msg))
.catch(err => console.error(err));
控制台打印:
- 收到连接。
- 使用连接完成!
- 完成被称为。
- 这应该是在连接关闭之后。