Promise 在 JavaScript 中很早就有各类的开源实现,ES6 将其归入了官方标准,提供了原生 api 支持,使用更加便捷。javascript
Promise 是一个对象,它用来标识 JavaScript 中异步操做的状态(pending, resolve, reject)及结果(data)。java
从控制台打印出来一个Promise 对象来看下
ajax
var p = new Promise((resolve, reject) => {
setTimeout(() => {
resolve('res.data');
}, 3000);
});
p.then((val) => {
console.log(val);
// 'res.data'
});
console.log(p);
// [object Promise]
Promise.resolve()方法能够将现有对象转为Promise 对象。api
var p = Promise.resolve($.ajax('/something.data'));
p.then((val) => {console.log(val)});
它等价于
var p = new Promise(resolve => {
resolve($.ajax('/something.data'))
});
p.then((val) => {console.log(val)});
此方法和Promise.resolve()方法相似,除了rejecet 表明状态为 Rejected,很少说。数组
用于将多个Promise 实例包装成一个新的 Promise实例,参数为一组 Promise 实例组成的数组。promise
var p = Promise.all([p1,p2,p3]);
当 p1, p2, p3 状态都 Resolved 的时候,p 的状态才会 Resolved;只要有一个实例 Rejected ,此时第一个被 Rejected 的实例的返回值就会传递给 P 的回调函数。app
应用场景:假设有一个接口,须要其它两个接口的数据做为参数,此时就要等那两个接口完成后才能执行请求。异步
var p1 = new Promise((resolve, reject) => {
setTimeout(resolve, 1000, 'P1');
});
var p2 = new Promise((resolve, reject) => {
setTimeout(resolve, 2000, 'P2');
});
// 同时执行p1和p2,并在它们都完成后执行then:
Promise.all([p1, p2]).then((results) => {
console.log(results); // 得到一个Array: ['P1', 'P2']
});
和Promise.all 相似,区别是 Promise.race() 只要监听到其中某一个实例改变状态,它的状态就跟着改变,并将那个改变状态实例的返回值传递给回调函数。async
应用场景: 能够经过多个异步任务来进行容错处理,多个接口返回一样的数据,只要有一个接口生效返回数据便可。函数
then 方法是定义在 Promise 的原型对象上的,做用是为 Promise 实例添加状态改变时的回调函数;
then() 返回一个新的Promise 实例,所以能够支持链式写法。
链式写法的一个例子//来自廖雪峰
// 0.5秒后返回input*input的计算结果:
function multiply(input) {
return new Promise((resolve, reject) => {
console.log('calculating ' + input + ' x ' + input + '...');
setTimeout(resolve, 500, input * input);
});
}
// 0.5秒后返回input+input的计算结果:
function add(input) {
return new Promise((resolve, reject) => {
console.log('calculating ' + input + ' + ' + input + '...');
setTimeout(resolve, 500, input + input);
});
}
var p = new Promise((resolve, reject) => {
console.log('start new Promise...');
resolve(123);
});
p.then(multiply)
.then(add)
.then(multiply)
.then(add)
.then(function (result) {
console.log('Got value: ' + result);
});
catch 方法是一个语法糖,看下面代码就明白了,用于指定发生错误时的回调函数。
var p = new Promise((resolve, rejecet) => {
if (...) {resolve()};
else {reject()};
})
p.then((val) => {
console.log('resolve:', val);
}).catch((err) => console.log('reject:', err));
// 等同于
p.then((data) => {
console.log(data);
}, (err) => {
console.log(err);
})
// 后一种写法更好,语义更清晰,第一种方法在第一个函数里面出错的话时在第二个函数里监听不到变化的。
实际开发中,常常遇到一种状况:不知道或者不想区分,函数 f 是同步函数仍是异步操做,可是想用 Promise 来处理它。由于这样就能够无论f是否包含异步操做,都用 then 方法指定下一步流程,用 catch 方法处理 f 抛出的错误。
const f = () => console.log('now');
Promise.resolve().then(f);
console.log('next');
// next
// now
const f = () => console.log('now');
(
() => new Promise(
resolve => resolve(f())
)
)();
console.log('next');
// now
// next
const f = () => console.log('now');
Promise.try(f);
console.log('next');
// now
// next
两个特色
感觉: