淺談Promise之參照Promise/A+規范實現Promise類

在需要多個操作的時候,會導致多個回調函數嵌套,導致代碼不夠直觀,就是常說的回調地獄,通常通過promise來解決

Promise本意是承諾,在程序中的意思就是承諾我過一段時間后會給你一個結果。 什么時候會用到過一段時間?答案是異步操作,異步是指可能比較長時間才有結果的才做,例如網絡請求、讀取本地文件等

按照Promise/A+規范來實現一個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);
  });
}
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • Promise 對象 Promise 的含義 Promise 是異步編程的一種解決方案,比傳統的解決方案——回調函...
    neromous閱讀 8,730評論 1 56
  • 本文適用的讀者 本文寫給有一定Promise使用經驗的人,如果你還沒有使用過Promise,這篇文章可能不適合你,...
    HZ充電大喵閱讀 7,335評論 6 19
  • Promiese 簡單說就是一個容器,里面保存著某個未來才會結束的事件(通常是一個異步操作)的結果,語法上說,Pr...
    雨飛飛雨閱讀 3,381評論 0 19
  • Promise的含義: ??Promise是異步編程的一種解決方案,比傳統的解決方案——回調函數和事件——更合理和...
    呼呼哥閱讀 2,190評論 0 16
  • 九月的重慶酷熱的嚴重,即使這是開學的季節。 夏清風一大早就被鬧鐘吵醒,習慣了暑假安逸的生活而現在又要開始努...
    貓很仙兒閱讀 163評論 0 1