Správný způsob, jak Přerušit (ukončit) běžící asynchronní/čeká na funkci?

0

Otázka

Tam byl další témata na SOBĚ, ale většina z nich jsou datovány před 5 lety. To, co je aktuální, up-to-date přístup zrušit čekají volání v JS? tj.

async myFunc(){
    let response = await oneHourLastingFunction();
    myProcessData(response);
}

v konkrétní okamžik aplikace rozhodne, že již nechcete čekat, že oneHourLastingFunction, ale to je uvízl v "čekají". Jak zrušit? Všechny standardní způsoby zrušení-žetony/abortControllers pro slibuje?

1

Nejlepší odpověď

1

Zrušení asynchronní postup je stále ještě není triviální úkol, a to zejména, když potřebujete hluboké zrušení a řízení toku. Neexistuje žádný nativní řešení v tuto chvíli. Vše, co můžete udělat nativně:

  • projít AbortController například do každé vnořené async funkci, kterou chcete, aby zrušit
  • přihlásit se všechny vnitřní mikro-úkolů (požadavků, časovače, atd.), aby signál
  • volitelně odhlášení úspěšně absolvováno mikro-úkolů ze signálu
  • volání abort metoda správce zrušit všechny objednané mikro-úkolů

Je to docela podrobné a složité řešení s potenciální nevracení paměti.

Mohu jen navrhnout vlastní řešení tohoto problému- c-promise2, který poskytuje zrušitelné přísliby a zrušitelné alternativa pro ECMA asynchronní funkce - generátory.

Zde je základní příklad (Live Demo):

import { CPromise } from "c-promise2";

// deeply cancelable generator-based asynchronous function
const oneHourLastingFunction = CPromise.promisify(function* () {
  // optionally just for logging
  this.onCancel(() =>
    console.log("oneHourLastingFunction::Cancel signal received")
  );
  yield CPromise.delay(5000); // this task will be cancelled on external timeout
  return "myData";
});

async function nativeAsyncFn() {
  await CPromise.delay(5000);
}

async function myFunc() {
  let response;
  try {
    response = await oneHourLastingFunction().timeout(2000);
  } catch (err) {
    if (!CPromise.isCanceledError(err)) throw err;
    console.warn("oneHourLastingFunction::timeout", err.code); // 'E_REASON_TIMEOUT'
  }
  await nativeAsyncFn(response);
}

const nativePromise = myFunc();

Hluboce zrušit roztoku (všechny funkce jsou zrušit) (Live Demo):

import { CPromise } from "c-promise2";

// deeply cancelable generator-based asynchronous function
const oneHourLastingFunction = CPromise.promisify(function* () {
  yield CPromise.delay(5000);
  return "myData";
});

const otherAsyncFn = CPromise.promisify(function* () {
  yield CPromise.delay(5000);
});

const myFunc = CPromise.promisify(function* () {
  let response;
  try {
    response = yield oneHourLastingFunction().timeout(2000);
  } catch (err) {
    if (err.code !== "E_REASON_TIMEOUT") throw err;
    console.log("oneHourLastingFunction::timeout");
  }
  yield otherAsyncFn(response);
});

const cancellablePromise = myFunc().then(
  (result) => console.log(`Done: ${result}`),
  (err) => console.warn(`Failed: ${err}`)
);

setTimeout(() => {
  console.log("send external cancel signal");
  cancellablePromise.cancel();
}, 4000);
2021-11-25 16:48:29

V jiných jazycích

Tato stránka je v jiných jazycích

Русский
..................................................................................................................
Italiano
..................................................................................................................
Polski
..................................................................................................................
Română
..................................................................................................................
한국어
..................................................................................................................
हिन्दी
..................................................................................................................
Français
..................................................................................................................
Türk
..................................................................................................................
Português
..................................................................................................................
ไทย
..................................................................................................................
中文
..................................................................................................................
Español
..................................................................................................................
Slovenský
..................................................................................................................