在需要多個操作的時候,會導致多個回撥函式巢狀,導致程式碼不夠直觀,就是常說的回撥地獄,通常通過promise來解決
Promise本意是承諾,在程式中的意思就是承諾我過一段時間後會給你一個結果。 什麼時候會用到過一段時間?答案是非同步操作,非同步是指可能比較長時間才有結果的才做,例如網路請求、讀取本地檔案等
建構函式初始化邏輯
const PENDING = 'pending';//初始態
const FULFILLED = 'fulfilled';//初始態
const REJECTED = 'rejected';//初始態
let self = this;//先快取當前promise例項
self.status = PENDING;//設定狀態
self.onResolvedCallbacks = [];//定義存放成功的回撥的陣列
self.onRejectedCallbacks = []; //定義存放失敗回撥的陣列
複製程式碼
executor執行器,包含兩個引數,分別是resolve 解決和reject 拒絕,new Promise這個executor就會執行
Promise有三個狀態:初始化狀態為pending,成功狀態為fulfilled,失敗狀態rejected,如果程式碼一旦成功就不會走向失敗,若 一直pending 永遠不給你明確的答覆
當呼叫以下方法的時候,如果promise狀態為pending的話可以轉成成功態,如果已經是成功態或者失敗態了,則什麼都不做
function resolve(value){
if(value!=null &&value.then&&typeof value.then == 'function'){
return value.then(resolve,reject);
}
setTimeout(function(){
if(self.status == PENDING){
self.status = FULFILLED;
self.value = value;
self.onResolvedCallbacks.forEach(cb=>cb(self.value));
}
})
}
function reject(reason){ //2.1.2
setTimeout(function(){
if(self.status == PENDING){
self.status = REJECTED;
self.value = reason;
self.onRejectedCallbacks.forEach(cb=>cb(self.value));
}
});
}
}
複製程式碼
因為此函式執行可能會異常,所以需要捕獲,如果出錯了,需要用錯誤物件reject,如果這函式執行失敗了,則用失敗的原因reject這個promise,需要用try...catch(e)...進行處理
try{
executor(resolve,reject);
}catch(e){
reject(e);
};
複製程式碼
Promise的解析過程
function resolvePromise(promise2,x,resolve,reject){
if(promise2 === x){
return reject(new TypeError('迴圈引用'));
}
let called = false;
if(x instanceof Promise){
if(x.status == PENDING){
x.then(function(y){
resolvePromise(promise2,y,resolve,reject);
},reject);
}else{
x.then(resolve,reject);
}
}else if(x!= null &&((typeof x=='object')||(typeof x == 'function'))){
try{
let then = x.then;
if(typeof then == 'function'){
then.call(x,function(y){
if(called)return;
called = true;
resolvePromise(promise2,y,resolve,reject)
},function(err){
if(called)return;
called = true;
reject(err);
});
}else{
resolve(x);
}
}catch(e){
if(called)return;
called = true;
reject(e);
}
}else{
resolve(x);
}
}
複製程式碼
then
方法就是用來指定Promise 物件的狀態改變時確定執行的操作,resolve 時執行第一個函式(onFulfilled),reject 時執行第二個函式(onRejected) 此方法中,如果成功和失敗的回撥沒有傳,則表示這個then沒有任何邏輯,只會把值往後拋
Promise.prototype.then = function(onFulfilled,onRejected){
onFulfilled = typeof onFulfilled == 'function'?onFulfilled:function(value){return value};
onRejected = typeof onRejected == 'function'?onRejected:reason=>{throw reason};
let self = this;
let promise2;
if(self.status == FULFILLED){
return promise2 = new Promise(function(resolve,reject){
setTimeout(function(){
try{
let x =onFulfilled(self.value);
resolvePromise(promise2,x,resolve,reject);
}catch(e){
reject(e);
}
})
});
}
if(self.status == REJECTED){
return promise2 = new Promise(function(resolve,reject){
setTimeout(function(){
try{
let x =onRejected(self.value);
resolvePromise(promise2,x,resolve,reject);
}catch(e){
reject(e);
}
})
});
}
if(self.status == PENDING){
return promise2 = new Promise(function(resolve,reject){
self.onResolvedCallbacks.push(function(){
try{
let x =onFulfilled(self.value);
resolvePromise(promise2,x,resolve,reject);
}catch(e){
reject(e);
}
});
self.onRejectedCallbacks.push(function(){
try{
let x =onRejected(self.value);
resolvePromise(promise2,x,resolve,reject);
}catch(e){
reject(e);
}
});
});
}
}
複製程式碼
promise的鏈式呼叫
- 每次呼叫返回的都是一個新的Promise例項
- 鏈式呼叫的引數通過返回值傳遞:即會將第一個then成功後,將他的返回值作為下一次成功的回撥函式的引數
then
可以使用鏈式呼叫的寫法原因在於,每一次執行該方法時總是會返回一個Promise
物件
catch
只是 promise.then(undefined, onRejected); 方法的一個別名而已。 也就是說,這個方法用來註冊當promise物件狀態變為Rejected時的回撥函式 catch原理就是隻傳失敗的回撥
Promise.prototype.catch = function(onRejected){
this.then(null,onRejected);
}
複製程式碼
Promise.all
接收一個 promise物件的陣列作為引數,當這個陣列裡的所有promise物件全部變為resolve或reject狀態的時候,它才會去呼叫 .then 方法
Promise.all = function(promises){
return new Promise(function(resolve,reject){
let done = gen(promises.length,resolve);
for(let i=0;i<promises.length;i++){
promises[i].then(function(data){
done(i,data);
},reject);
}
});
}
複製程式碼
Promise.race
只要有一個promise物件進入 FulFilled 或者 Rejected 狀態的話,就會繼續進行後面的處理
Promise.race = function(promises){
return new Promise(function(resolve,reject){
for(let i=0;i<promises.length;i++){
promises[i].then(resolve,reject);
}
});
}
複製程式碼
別人提供 給你一個方法,需要你傳入一個promise,但你只有一個普通的值,你就可以通過這個方法把這個普通的值(string number object)轉成一個promise物件 返回一個立刻成功的promise
Promise.resolve = function(value){
return new Promise(function(resolve){
resolve(value);
});
}
複製程式碼
返回一個立刻失敗的promise
Promise.reject = function(reason){
return new Promise(function(resolve,reject){
reject(reason);
});
}
複製程式碼