欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

JavaScript 中斷請求幾種方案詳解

 更新時間:2021年09月09日 14:37:27   作者:想成為大佬的鴨子  
這篇文章主要介紹了JavaScript 中斷請求幾種方案詳解,本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下

1 Promise

Promise有一個缺點是一旦創(chuàng)建無法取消,所以本質(zhì)上Promise是無法被終止的.

但是我們可以通過中斷調(diào)用鏈或中斷Promise來模擬請求的中斷.

中斷調(diào)用鏈

中斷調(diào)用鏈就是在某一個then/catch執(zhí)行之后,后續(xù)的鏈?zhǔn)秸{(diào)用(包括then,catch,finally)不再繼續(xù)執(zhí)行.

方法是在then/catch返回一個新的Promise實例,并保持pending狀態(tài):

new Promise((resolve, reject) => {
    setTimeout(() => {
        resolve('result');
    });
}).then(res => {
    // 達(dá)到某種條件,return一個pending狀態(tài)的Promise實例,以中斷調(diào)用鏈
    if (res === 'result') {
        return new Promise(() => {});
    }
    console.log(res); // 不打印
}).then(() => {
    console.log('then不執(zhí)行'); // 不打印
}).catch(() => {
    console.log('catch不執(zhí)行'); // 不打印
}).finally(() => {
    console.log('finally不執(zhí)行'); // 不打印
});

中斷Promise

中斷Promise不等同于中止Promise,因為Promise是無法被終止的.

這里的中斷指的是,在合適的時機,把pending狀態(tài)的promise給reject掉.例如一個常見的應(yīng)用場景就是給網(wǎng)絡(luò)請求設(shè)置超時時間,一旦超時就中斷.

老規(guī)矩,用setTimeout來模擬網(wǎng)絡(luò)請求.閥值設(shè)置為Math.random() * 3000表示隨機3秒之內(nèi)返回結(jié)果.

const request = new Promise((resolve, reject) => {
  setTimeout(() => {
    resolve('收到服務(wù)端數(shù)據(jù)')
  }, Math.random() * 3000)
})

假設(shè)超過2秒就是網(wǎng)絡(luò)超時,我們可以封裝一個超時處理函數(shù).

由于網(wǎng)絡(luò)請求所需的事件是隨機的,因此可以利用Promise.race方法,達(dá)到超時reject的目的.

const timeoutReject = (p1, timeout = 2000) => {
    const p2 = new Promise((resolve, reject) => {
        setTimeout(() => {
            reject('網(wǎng)絡(luò)超時');
        }, timeout);
    });
    return Promise.race([p1, p2]);
};

timeoutReject(request).then(res => {
    console.log(res);
}).catch(err => {
    console.log(err);
});

包裝abort方法——仿照Axios的CancelToken

上面實現(xiàn)的方式并不靈活,因為中斷Promise的方式有很多,不單單是網(wǎng)絡(luò)超時.

我們可以仿照Axios中CancelToken的核心源碼,簡單包裝一個abort方法,供使用者隨時調(diào)用.

function abortWrapper(p1) {
    let abort;
    const p2 = new Promise((resolve, reject) => {
        abort = reject;
    });
    // 如果沒有resolve或reject,p2的狀態(tài)永遠(yuǎn)是pending
    const p = Promise.race([p1, p2]);
    p.abort = abort;
    return p;
}

const req = abortWrapper(request);
req.then(res => {
    console.log(res);
}).catch(err => {
    console.log(err);
});

setTimeout(() => {
    // 手動調(diào)用req.abort,將p2的狀態(tài)改變?yōu)閞ejected
    req.abort('手動中斷請求');
}, 2000);

如此封裝的主要目的就是為了能夠在Promise外部控制其resolve或reject,讓使用者可以隨時手動調(diào)用resolve(觸發(fā).then)或reject(觸發(fā).catch).

需要注意的是,雖然Promise請求被中斷了,但是promise并沒有終止,網(wǎng)絡(luò)請求依然可能返回,只不過那時我們已經(jīng)不關(guān)心請求結(jié)果了.

2 RXJS的unsubscribe方法

rxjs本身提供了取消訂閱的方法,即unsubscribe.

let stream1$ = new Observable(observer => {
    let timeout = setTimeout(() => {
        observer.next('observable timeout');
    }, 2000);

    return () => {
        clearTimeout(timeout);
    }
});
let disposable = stream1$.subscribe(value => console.log(value));
setTimeout(() => {
    disposable.unsubscribe();
}, 1000);

3 Axios的CancelToken

Axios的CancelToken有兩種使用方法:

  • 方法一
import axios from 'axios';
const CancelToken = axios.CancelToken;
const source = CancelToken.source();

axios.get('/user/12345', {
  cancelToken: source.token
}).catch(function (thrown) {
  if (axios.isCancel(thrown)) {
    console.log('Request canceled', thrown.message);
  } else {
    // handle error
  } 
});

source.cancel('Operation canceled by the user.');
  • 方法二
import axios from 'axios';
const CancelToken = axios.CancelToken;

// 創(chuàng)建一個變量如 cancel 用于存儲這個中斷某個請求的方法
let cancel;

axios.get('/user/12345', {
  cancelToken: new CancelToken(function executor(c) {
    cancel = c; // 將參數(shù) c 賦值給 cancel
  })
});

// 判斷 cancel 是否為函數(shù),確保 axios 已實例化一個CancelToken
if (typeof cancel === 'function') {
    cancel();
    cancel = null;
}

CancelToken的核心源碼:(axios/lib/cancel/CancelToken.js)

'use strict';

var Cancel = require('./Cancel');

/**
 * A `CancelToken` is an object that can be used to request cancellation of an operation.
 *
 * @class
 * @param {Function} executor The executor function.
 */
function CancelToken(executor) {
  if (typeof executor !== 'function') {
    throw new TypeError('executor must be a function.');
  }

  var resolvePromise;
  this.promise = new Promise(function promiseExecutor(resolve) {
    resolvePromise = resolve;
  });

  var token = this;
  executor(function cancel(message) {
    if (token.reason) {
      // Cancellation has already been requested
      return;
    }

    token.reason = new Cancel(message);
    resolvePromise(token.reason);
  });
}

/**
 * Throws a `Cancel` if cancellation has been requested.
 */
CancelToken.prototype.throwIfRequested = function throwIfRequested() {
  if (this.reason) {
    throw this.reason;
  }
};

/**
 * Returns an object that contains a new `CancelToken` and a function that, when called,
 * cancels the `CancelToken`.
 */
CancelToken.source = function source() {
  var cancel;
  var token = new CancelToken(function executor(c) {
    cancel = c;
  });
  return {
    token: token,
    cancel: cancel
  };
};

module.exports = CancelToken;

可以看到,在Axios底層,CancelToken的核心源碼所體現(xiàn)的思想,與上面中斷Promise包裝abort方法的思想一致.

只不過Axios在外部手動調(diào)用resolve(用戶觸發(fā)cancel方法),而resolve一旦調(diào)用,就會觸發(fā)promise的then方法,來看這個promise.then的源碼:(axios/lib/adapters/xhr.js)

if (config.cancelToken) {
  // Handle cancellation
  config.cancelToken.promise.then(function onCanceled(cancel) {
    if (!request) {
      return;
    }

    request.abort();
    reject(cancel);
    // Clean up request
    request = null;
  });
}

可以看到then方法中會執(zhí)行abort方法取消請求,同時調(diào)用reject讓外層的promise失敗.

到此這篇關(guān)于JavaScript 中斷請求幾種方案詳解的文章就介紹到這了,更多相關(guān)js中斷請求內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 移動端網(wǎng)頁開發(fā)調(diào)試神器Eruda的介紹與使用技巧

    移動端網(wǎng)頁開發(fā)調(diào)試神器Eruda的介紹與使用技巧

    在日常的移動端開發(fā)時,一般都是試用chrome瀏覽器的移動端模式進(jìn)行開發(fā)和調(diào)試,只有在chrome調(diào)試完成,而最近發(fā)現(xiàn)了一個新的調(diào)試方法,所以這篇文章主要給大家介紹了關(guān)于移動端網(wǎng)頁開發(fā)調(diào)試神器Eruda的基本資料,以及其使用的一些技巧,需要的朋友可以參考下。
    2017-10-10
  • 一個層慢慢增高展開,有種向下滑動的效果

    一個層慢慢增高展開,有種向下滑動的效果

    一個層慢慢增高展開,有種向下滑動的效果...
    2006-11-11
  • javascript檢查瀏覽器是否支持flash的實現(xiàn)代碼

    javascript檢查瀏覽器是否支持flash的實現(xiàn)代碼

    這篇文章主要介紹了javascript檢查瀏覽器是否支持flash的實現(xiàn)代碼,需要的朋友可以參考下
    2014-08-08
  • js實現(xiàn)移動端編輯添加地址【模仿京東】

    js實現(xiàn)移動端編輯添加地址【模仿京東】

    本篇文章主要介紹了js實現(xiàn)移動端編輯添加地址[模仿京東]的實例。具有很好的參考價值。下面跟著小編一起來看下吧
    2017-04-04
  • taro小程序添加骨架屏的實現(xiàn)代碼

    taro小程序添加骨架屏的實現(xiàn)代碼

    這篇文章主要介紹了taro小程序添加骨架屏的實現(xiàn)代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-11-11
  • JavaScript 拾漏補遺

    JavaScript 拾漏補遺

    javascritp實際上由三部分組成: ECMAScript,DOM, BOM 前兩者由工業(yè)標(biāo)準(zhǔn),BOM尚比較混亂。
    2009-12-12
  • 用JavaScrpt實現(xiàn)文件夾簡單輕松加密的實現(xiàn)方法圖文

    用JavaScrpt實現(xiàn)文件夾簡單輕松加密的實現(xiàn)方法圖文

    電腦里經(jīng)常會存儲著重要文件,這些文件需要進(jìn)行加密,有許多方法來實現(xiàn)。但如果想對一個文件夾里的所有文件都進(jìn)行加密,數(shù)量少還可以,要是數(shù)量多豈不是得把人累死?
    2008-09-09
  • 原生JS實現(xiàn)煙花效果

    原生JS實現(xiàn)煙花效果

    這篇文章主要為大家詳細(xì)介紹了原生JS實現(xiàn)煙花效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-03-03
  • ES6 對象的新功能與解構(gòu)賦值介紹

    ES6 對象的新功能與解構(gòu)賦值介紹

    這篇文章主要介紹了ES6 對象的新功能與解構(gòu)賦值介紹,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2019-02-02
  • JavaScript中Number對象的toFixed() 方法詳解

    JavaScript中Number對象的toFixed() 方法詳解

    下面小編就為大家?guī)硪黄狫avaScript中Number對象的toFixed() 方法詳解。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-09-09

最新評論