标签:success 并行执行 需要 指定 错误 rop 报错 HERE xxx
|
1
2
3
4
5
6
7
8
9
10
11
12
|
firstAsync(function(data){ //处理得到的 data 数据 //.... secondAsync(function(data2){ //处理得到的 data2 数据 //.... thirdAsync(function(data3){ //处理得到的 data3 数据 //.... }); });}); |
(2)如果使用 promises 的话,代码就会变得扁平且更可读了。前面提到 then 返回了一个 promise,因此我们可以将 then 的调用不停地串连起来。其中 then 返回的 promise 装载了由调用返回的值。
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
firstAsync().then(function(data){ //处理得到的 data 数据 //.... return secondAsync();}).then(function(data2){ //处理得到的 data2 数据 //.... return thirdAsync();}).then(function(data3){ //处理得到的 data3 数据 //....}); |
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
function fetch(callback) { setTimeout(() => { throw Error(‘请求失败‘) }, 2000)}try { fetch(() => { console.log(‘请求处理‘) // 永远不会执行 })} catch (error) { console.log(‘触发异常‘, error) // 永远不会执行}// 程序崩溃// Uncaught Error: 请求失败 |
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
function fetch(callback) { return new Promise((resolve, reject) => { setTimeout(() => { reject(‘请求失败‘); }, 2000) })}fetch().then( function(data){ console.log(‘请求处理‘); console.log(data); }, function(reason, data){ console.log(‘触发异常‘); console.log(reason); }); |
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
function fetch(callback) { return new Promise((resolve, reject) => { setTimeout(() => { reject(‘请求失败‘); }, 2000) })}fetch().then( function(data){ console.log(‘请求处理‘); console.log(data); }).catch(function(reason){ console.log(‘触发异常‘); console.log(reason);}); |
第二部分:
2015年6月, ES2015(即 ECMAScript 6、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 链式调用这三个方法:
|
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 + ‘没法吃!‘);}) |
运行结果如下:
|
1
2
3
4
|
cook().then(null, function(data){ console.log(data + ‘没法吃!‘);}) |
(1)它可以和 then 的第二个参数一样,用来指定 reject 的回调
|
1
2
3
4
5
|
cook().then(eat).catch(function(data){ console.log(data + ‘没法吃!‘);}); |
(2)它的另一个作用是,当执行 resolve 的回调(也就是上面 then 中的第一个参数)时,如果抛出异常了(代码出错了),那么也不会报错卡死 js,而是会进到这个 catch 方法中。
|
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);}); |
运行结果如下:
|
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 中赛跑。
第三部分:
上文我介绍了 ES6 中的 Promise,它完全遵循 Promises/A 规范。而我们熟悉的 jQuery 又有自己的 Promise 实现:Deferred(但其并不是遵循 Promises/A 规范)。本文就讲讲 jQuery 中 Promise 的实现。
下面我们定义做饭、吃饭、洗碗(cook、eat、wash)这三个方法(这里使用 setTimeout 模拟异步操作)
|
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
35
|
//做饭function cook(){ console.log(‘开始做饭。‘); var def = $.Deferred(); //执行异步操作 setTimeout(function(){ console.log(‘做饭完毕!‘); def.resolve(‘鸡蛋炒饭‘); }, 1000); return def.promise();}//吃饭function eat(data){ console.log(‘开始吃饭:‘ + data); var def = $.Deferred(); //执行异步操作 setTimeout(function(){ console.log(‘吃饭完毕!‘); def.resolve(‘一块碗和一双筷子‘); }, 1000); return def.promise();}//洗碗function wash(data){ console.log(‘开始洗碗:‘ + data); var def = $.Deferred(); //执行异步操作 setTimeout(function(){ console.log(‘洗碗完毕!‘); def.resolve(‘干净的碗筷‘); }, 1000); return def.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);}); |
(2)运行结果如下:
|
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 def = $.Deferred(); //执行异步操作 setTimeout(function(){ console.log(‘做饭完毕!‘); def.reject(‘烧焦的米饭‘); }, 1000); return def.promise();}//吃饭function eat(data){ console.log(‘开始吃饭:‘ + data); var def = $.Deferred(); //执行异步操作 setTimeout(function(){ console.log(‘吃饭完毕!‘); def.resolve(‘一块碗和一双筷子‘); }, 1000); return def.promise();}cook().then(eat, function(data){ console.log(data + ‘没法吃!‘);}) |
运行结果如下:
|
1
|
deferred.then( doneFilter [, failFilter ] [, progressFilter ] ) |
done 和 fail 是 jQuery 增加的两个语法糖方法。分别用来指定执行完成和执行失败的回调。
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
//then方法d.then(function(){ console.log(‘执行完成‘);}, function(){ console.log(‘执行失败‘);});//done方法、fail方法d.done(function(){ console.log(‘执行完成‘);}).fail(function(){ console.log(‘执行失败‘);}); |
jQuery 的 Deferred 对象上还有一个 always 方法,不论执行完成还是执行失败,always 都会执行,有点类似 ajax 中的 complete。
|
1
2
3
4
5
6
|
cook().then(eat).then(wash).always(function(){ console.log(‘上班去!‘);}) |
在开头讲到,目前 Promise 事实上的标准是社区提出的 Promises/A 规范,jQuery 的实现并不完全符合 Promises/A,主要表现在对错误的处理不同。
|
1
2
3
4
5
6
7
8
|
cook().then(function(data){ throw new Error(‘米饭被打翻了!‘); eat(data);}).catch(function(data){ console.log(data);}); |
同样我们在回调函数中抛出一个错误,jQuery 的 Deferred 对象此时不会改变状态,亦不会触发回调函数,该错误一般情况下会被 window.onerror 捕获。换句话说,在 Deferred 对象中,总是必须使用 reject 方法来改变状态。
|
1
2
3
4
5
6
7
8
9
10
|
cook().then(function(data){ throw new Error(‘米饭被打翻了!‘); eat(data);})window.onerror = function(msg, url, line) { console.log("发生错误了:" + msg); return true; //如果注释掉该语句,浏览器中还是会有错误提示,反之则没有。} |
|
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
|
//切菜function cutUp(){ console.log(‘开始切菜。‘); var def = $.Deferred(); //执行异步操作 setTimeout(function(){ console.log(‘切菜完毕!‘); def.resolve(‘切好的菜‘); }, 1000); return def.promise();}//烧水function boil(){ console.log(‘开始烧水。‘); var def = $.Deferred(); //执行异步操作 setTimeout(function(){ console.log(‘烧水完毕!‘); def.resolve(‘烧好的水‘); }, 1000); return def.promise();}$.when(cutUp(), boil()).then(function(data1, data2){ console.log("准备工作完毕:"); console.log(data1, data2);}); |
jQuery 中我们常常会用到的 ajax, get, post 等 Ajax 函数,其实它们内部都已经实现了 Deferred。这些方法调用后会返回一个受限的 Deferred 对象。既然是 Deferred 对象,那么自然也有上面提到的所有特性。
|
1
2
3
4
5
6
7
8
9
10
11
12
13
|
req1 = function(){ return $.ajax(/*...*/);}req2 = function(){ return $.ajax(/*...*/);}req3 = function(){ return $.ajax(/*...*/);}req1().then(req2).then(req3).done(function(){ console.log(‘请求发送完毕‘);}); |
success、error、complete是 ajax 提供的语法糖,功能与 Deferred 对象的 done、fail、always 一致。比如下面两段代码功能是一致的:
|
1
2
3
4
5
6
7
8
9
10
11
|
//使用success、error、complete$.ajax(/*...*/).success(function(){/*...*/}).error(function(){/*...*/}).complete(function(){/*...*/})//使用done、fail、always$.ajax(/*...*/).done(function(){/*...*/}).fai(function(){/*...*/}).always(function(){/*...*/}) |
标签:success 并行执行 需要 指定 错误 rop 报错 HERE xxx
原文地址:https://www.cnblogs.com/cx19950223/p/11049880.html