成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

淺談Promise之參照Promise/A+規(guī)范實現(xiàn)Promise類

roundstones / 788人閱讀

摘要:在需要多個操作的時候,會導(dǎo)致多個回調(diào)函數(shù)嵌套,導(dǎo)致代碼不夠直觀,就是常說的回調(diào)地獄,通常通過來解決本意是承諾,在程序中的意思就是承諾我過一段時間后會給你一個結(jié)果。

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

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

按照Promise/A+規(guī)范來實現(xiàn)一個Promise類
構(gòu)造函數(shù)初始化邏輯
const PENDING =  "pending";//初始態(tài)
const FULFILLED =  "fulfilled";//初始態(tài)
const REJECTED =  "rejected";//初始態(tài)
let self = this;//先緩存當(dāng)前promise實例
self.status = PENDING;//設(shè)置狀態(tài)
self.onResolvedCallbacks = [];//定義存放成功的回調(diào)的數(shù)組
self.onRejectedCallbacks = []; //定義存放失敗回調(diào)的數(shù)組
executor執(zhí)行器,包含兩個參數(shù),分別是resolve 解決和reject 拒絕,new Promise這個executor就會執(zhí)行

Promise有三個狀態(tài):初始化狀態(tài)為pending,成功狀態(tài)為fulfilled,失敗狀態(tài)rejected,如果代碼一旦成功就不會走向失敗,若 一直pending 永遠(yuǎn)不給你明確的答復(fù)

當(dāng)調(diào)用以下方法的時候,如果promise狀態(tài)為pending的話可以轉(zhuǎn)成成功態(tài),如果已經(jīng)是成功態(tài)或者失敗態(tài)了,則什么都不做

  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));
      }
    });
  }
}
因為此函數(shù)執(zhí)行可能會異常,所以需要捕獲,如果出錯了,需要用錯誤對象reject,如果這函數(shù)執(zhí)行失敗了,則用失敗的原因reject這個promise,需要用try...catch(e)...進(jìn)行處理
  try{
    executor(resolve,reject);
  }catch(e){
    reject(e);
  };
Promise的解析過程
function resolvePromise(promise2,x,resolve,reject){
  if(promise2 === x){
    return reject(new TypeError("循環(huán)引用"));
  }
  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 對象的狀態(tài)改變時確定執(zhí)行的操作,resolve 時執(zhí)行第一個函數(shù)(onFulfilled),reject 時執(zhí)行第二個函數(shù)(onRejected)
此方法中,如果成功和失敗的回調(diào)沒有傳,則表示這個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的鏈?zhǔn)秸{(diào)用

每次調(diào)用返回的都是一個新的Promise實例

鏈?zhǔn)秸{(diào)用的參數(shù)通過返回值傳遞:即會將第一個then成功后,將他的返回值作為下一次成功的回調(diào)函數(shù)的參數(shù)

then可以使用鏈?zhǔn)秸{(diào)用的寫法原因在于,每一次執(zhí)行該方法時總是會返回一個Promise對象

catch 只是 promise.then(undefined, onRejected); 方法的一個別名而已。 也就是說,這個方法用來注冊當(dāng)promise對象狀態(tài)變?yōu)镽ejected時的回調(diào)函數(shù)
catch原理就是只傳失敗的回調(diào)
Promise.prototype.catch = function(onRejected){
  this.then(null,onRejected);
}
Promise.all 接收一個 promise對象的數(shù)組作為參數(shù),當(dāng)這個數(shù)組里的所有promise對象全部變?yōu)閞esolve或reject狀態(tài)的時候,它才會去調(diào)用 .then 方法
Promise.all = function(promises){
 return new Promise(function(resolve,reject){
   let done = gen(promises.length,resolve);
   for(let i=0;i
Promise.race只要有一個promise對象進(jìn)入 FulFilled 或者 Rejected 狀態(tài)的話,就會繼續(xù)進(jìn)行后面的處理
Promise.race = function(promises){
  return new Promise(function(resolve,reject){
    for(let i=0;i
別人提供 給你一個方法,需要你傳入一個promise,但你只有一個普通的值,你就可以通過這個方法把這個普通的值(string number object)轉(zhuǎn)成一個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);
  });
}

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/92494.html

相關(guān)文章

  • 3行代碼實現(xiàn)一個簡易版promise

    前言 作為一個后端過來的同學(xué),剛?cè)腴T前端的時候,被js種種「反人類」的概念折騰的死去活來的.其中一個印象比較深刻的,就是promise,感覺實在太難理解了...所有就有了寫個簡單的promise的想法.希望能幫助到一些跟我一樣,感覺promise很難理解的新同學(xué). promise的教程網(wǎng)上多如牛毛,其中寫的比較通俗易懂的莫過于阮一峰的es6,反正我是他的書才懂的.所以今天,我們也不會來復(fù)述一遍如何...

    ralap 評論0 收藏0
  • 淺談Javascript中Promise對象的實現(xiàn)

    摘要:我們可以進(jìn)行適當(dāng)?shù)母倪M(jìn),把回調(diào)函數(shù)寫到外面即使是改寫成這樣,代碼還是不夠直觀,但是如果有了對象,代碼就可以寫得非常清晰,一目了然,請看這樣函數(shù)就不用寫在的回調(diào)中了目前的標(biāo)準(zhǔn)中還未支持對象,那么我們就自己動手,豐衣足食吧。 本文同步自我得博客:http://www.joeray61.com 很多做前端的朋友應(yīng)該都聽說過Promise(或者Deferred)對象,今天我就講一下我對Prom...

    caiyongji 評論0 收藏0
  • 淺談ES6原生Promise

    摘要:如果有錯誤,則到的第二個回調(diào)函數(shù)中,對錯誤進(jìn)行處理。假設(shè)第一個的第一個回調(diào)沒有返回一個對象,那么第二個的調(diào)用者還是原來的對象,只不過其的值變成了第一個中第一個回調(diào)函數(shù)的返回值。 ES6標(biāo)準(zhǔn)出爐之前,一個幽靈,回調(diào)的幽靈,游蕩在JavaScript世界。 正所謂: 世界本沒有回調(diào),寫的人多了,也就有了})})})})})。 Promise的興起,是因為異步方法調(diào)用中,往往會出現(xiàn)回調(diào)函數(shù)一...

    yedf 評論0 收藏0
  • JavaScript Promise啟示錄

    摘要:近幾年隨著開發(fā)模式的逐漸成熟,規(guī)范順勢而生,其中就包括提出了規(guī)范,完全改變了異步編程的寫法,讓異步編程變得十分的易于理解。最后,是如此的優(yōu)雅但也只是解決了回調(diào)的深層嵌套的問題,真正簡化異步編程的還是,在端,建議考慮。 本篇,簡單實現(xiàn)一個promise,主要普及promise的用法。 一直以來,JavaScript處理異步都是以callback的方式,在前端開發(fā)領(lǐng)域callback機(jī)制...

    Juven 評論0 收藏0
  • [轉(zhuǎn)載·JS] JavaScript Promise啟示錄

    摘要:近幾年隨著開發(fā)模式的逐漸成熟,規(guī)范順勢而生,其中就包括提出了規(guī)范,完全改變了異步編程的寫法,讓異步編程變得十分的易于理解。最后,是如此的優(yōu)雅但也只是解決了回調(diào)的深層嵌套的問題,真正簡化異步編程的還是,在端,建議考慮。 前段時間頻頻看到Promise這個詞,今天發(fā)現(xiàn)騰訊AlloyTeam寫得這篇很贊,遂轉(zhuǎn)之。 原文鏈接 本篇,主要普及promise的用法。 一直以來,JavaScrip...

    Lyux 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<