原文地址:https://www.hangge.com/blog/cache/detail_1638.htmlhtml
2015年6月, ES2015(即 ECMAScript 六、ES6) 正式發佈。其中 Promise 被列爲正式規範,成爲 ES6 中最重要的特性之一。數組
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
|
//作飯
function
cook(){
console.log(
'開始作飯。'
);
var
p =
new
Promise(
function
(resolve, reject){
//作一些異步操做
setTimeout(
function
(){
console.log(
'作飯完畢!'
);
resolve(
'雞蛋炒飯'
);
}, 1000);
});
return
p;
}
//吃飯
function
eat(data){
console.log(
'開始吃飯:'
+ data);
var
p =
new
Promise(
function
(resolve, reject){
//作一些異步操做
setTimeout(
function
(){
console.log(
'吃飯完畢!'
);
resolve(
'一塊碗和一雙筷子'
);
}, 2000);
});
return
p;
}
function
wash(data){
console.log(
'開始洗碗:'
+ data);
var
p =
new
Promise(
function
(resolve, reject){
//作一些異步操做
setTimeout(
function
(){
console.log(
'洗碗完畢!'
);
resolve(
'乾淨的碗筷'
);
}, 2000);
});
return
p;
}
|
(2)使用 then 鏈式調用這三個方法:promise
1
2
3
4
5
6
7
8
9
10
|
cook()
.then(
function
(data){
return
eat(data);
})
.then(
function
(data){
return
wash(data);
})
.then(
function
(data){
console.log(data);
});
|
固然上面代碼還能夠簡化成以下:異步
1
2
3
4
5
6
|
cook()
.then(eat)
.then(wash)
.then(
function
(data){
console.log(data);
});
|
(3)運行結果以下:函數
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
//作飯
function
cook(){
console.log(
'開始作飯。'
);
var
p =
new
Promise(
function
(resolve, reject){
//作一些異步操做
setTimeout(
function
(){
console.log(
'作飯失敗!'
);
reject(
'燒焦的米飯'
);
}, 1000);
});
return
p;
}
//吃飯
function
eat(data){
console.log(
'開始吃飯:'
+ data);
var
p =
new
Promise(
function
(resolve, reject){
//作一些異步操做
setTimeout(
function
(){
console.log(
'吃飯完畢!'
);
resolve(
'一塊碗和一雙筷子'
);
}, 2000);
});
return
p;
}
cook()
.then(eat,
function
(data){
console.log(data +
'無法吃!'
);
})
|
運行結果以下:spa
1
2
3
4
|
cook()
.then(
null
,
function
(data){
console.log(data +
'無法吃!'
);
})
|
(1)它能夠和 then 的第二個參數同樣,用來指定 reject 的回調rest
1
2
3
4
5
|
cook()
.then(eat)
.
catch
(
function
(data){
console.log(data +
'無法吃!'
);
});
|
(2)它的另外一個做用是,當執行 resolve 的回調(也就是上面 then 中的第一個參數)時,若是拋出異常了(代碼出錯了),那麼也不會報錯卡死 js,而是會進到這個 catch 方法中。code
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
|
//作飯
function
cook(){
console.log(
'開始作飯。'
);
var
p =
new
Promise(
function
(resolve, reject){
//作一些異步操做
setTimeout(
function
(){
console.log(
'作飯完畢!'
);
resolve(
'雞蛋炒飯'
);
}, 1000);
});
return
p;
}
//吃飯
function
eat(data){
console.log(
'開始吃飯:'
+ data);
var
p =
new
Promise(
function
(resolve, reject){
//作一些異步操做
setTimeout(
function
(){
console.log(
'吃飯完畢!'
);
resolve(
'一塊碗和一雙筷子'
);
}, 2000);
});
return
p;
}
cook()
.then(
function
(data){
throw
new
Error(
'米飯被打翻了!'
);
eat(data);
})
.
catch
(
function
(data){
console.log(data);
});
|
運行結果以下:htm
1
2
3
4
5
6
7
8
9
10
11
|
somePromise.then(
function
() {
return
a();
}).
catch
(TypeError,
function
(e) {
//If a is defined, will end up here because
//it is a type error to reference property of undefined
}).
catch
(ReferenceError,
function
(e) {
//Will end up here if a wasn't defined at all
}).
catch
(
function
(e) {
//Generic catch-the rest, error wasn't TypeError nor
//ReferenceError
});
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
//切菜
function
cutUp(){
console.log(
'開始切菜。'
);
var
p =
new
Promise(
function
(resolve, reject){
//作一些異步操做
setTimeout(
function
(){
console.log(
'切菜完畢!'
);
resolve(
'切好的菜'
);
}, 1000);
});
return
p;
}
//燒水
function
boil(){
console.log(
'開始燒水。'
);
var
p =
new
Promise(
function
(resolve, reject){
//作一些異步操做
setTimeout(
function
(){
console.log(
'燒水完畢!'
);
resolve(
'燒好的水'
);
}, 1000);
});
return
p;
}
Promise
.all([cutUp(), boil()])
.then(
function
(results){
console.log(
"準備工做完畢:"
);
console.log(results);
});
|
(2)運行結果以下:對象
1
2
3
4
5
6
|
Promise
.race([cutUp(), boil()])
.then(
function
(results){
console.log(
"準備工做完畢:"
);
console.log(results);
});
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
//請求某個圖片資源
function
requestImg(){
var
p =
new
Promise(
function
(resolve, reject){
var
img =
new
Image();
img.onload =
function
(){
resolve(img);
}
img.src =
'xxxxxx'
;
});
return
p;
}
//延時函數,用於給請求計時
function
timeout(){
var
p =
new
Promise(
function
(resolve, reject){
setTimeout(
function
(){
reject(
'圖片請求超時'
);
}, 5000);
});
return
p;
}
Promise
.race([requestImg(), timeout()])
.then(
function
(results){
console.log(results);
})
.
catch
(
function
(reason){
console.log(reason);
});
|
上面代碼 requestImg 函數異步請求一張圖片,timeout 函數是一個延時 5 秒的異步操做。咱們將它們一塊兒放在 race 中賽跑。