I gestori delle Promise .then/.catch/.finally sono sempre asincroni.
Anche quando una Promise è immediatamente risolta, il codice sulle linee sotto .then/.catch/.finally verrà sempre eseguito prima dei gestori.
Ecco una dimostrazione:
let promise = Promise.resolve();
promise.then(() => alert("promise completa"));
alert("codice finito"); // questo alert viene mostrato prima
Se lo esegui, vedrai prima codice finito, in seguito promise done.
Questo è strano, perché la Promise è chiaramente completa dallâinizio.
Perché quindi il .then viene eseguito dopo? Cosa succede?
Coda dei Microtask (Microtasks Queue)
I task asincroni hanno bisogno di una gestione appropriata. Per questo motivo, lo standard specifica una coda interna PromiseJobs, più spesso riferita come âcoda dei microtaskâ (microtask queue) (termine di v8).
Come detto nella specifica:
- La coda è primo-dentro-primo-fuori: i task messi in coda per primi sono eseguiti per primi.
- Lâesecuzione di un task è iniziata solo quando nientâaltro è in esecuzione.
Oppure, per dirla in modo semplice, quando una promise è pronta, i suoi gestori .then/catch/finally sono messi nella coda. Non vengono ancora eseguiti. Il motore JavaScript prende un task dalla coda e lo esegue, quando diventa libero dal codice corrente.
Questo è il motivo per cui âcodice finitoâ nellâesempio sopra viene mostrato prima.
I gestori delle promise passano sempre da quella coda interna.
Se câè una catena con diversi .then/catch/finally, allora ognuno di essi viene eseguito in modo asincrono. Cioè, viene prima messo in coda ed eseguito quando il codice corrente è completo e i gestori messi in coda precedentemente sono finiti.
Che cosa succede se per noi lâordine è importante? Come possiamo far funzionare code finished dopo promise done?
Facile, basta metterlo in coda con .then:
Promise.resolve()
.then(() => alert("promise done!"))
.then(() => alert("code finished"));
Ora lâordine è come inteso.
Rigetto non gestito (Unhandled rejection)
Ricordi lâevento âunhandledrejectionâ dal capitolo Gestione degli errori con le promise?
Ora possiamo vedere esattamente come JavaScript viene a conoscenza che câè stato un respingimento non gestito (unhandled rejection)
âUnhandled rejectionâ avviene quando un errore di una promise non è gestito alla fine della coda dei microtask
Normalmente, se ci aspettiamo un errore, aggiungiamo .catch alla catena delle promise per gestirlo:
let promise = Promise.reject(new Error("Promise Fallita!"));
promise.catch(err => alert('catturato'));
// non viene eseguito: errore gestito
window.addEventListener('unhandledrejection', event => alert(event.reason));
â¦Ma se ci dimentichiamo di aggiungere .catch, allora, dopo che la coda dei microtask è vuota, il motore innesca lâevento:
let promise = Promise.reject(new Error("Promise Fallita!"));
// Promise Fallita!
window.addEventListener('unhandledrejection', event => alert(event.reason));
Cosa succede se gestiamo lâerrore dopo? Come qui:
let promise = Promise.reject(new Error("Promise Fallita!"));
setTimeout(() => promise.catch(err => alert('caught')), 1000);
// Error: Promise Fallita!
window.addEventListener('unhandledrejection', event => alert(event.reason));
Ora il respingimento non gestito appare di nuovo. Perché? unhandledrejection viene innescato quando la coda dei microtask è completa. Il motore esamina le promise e, se qualcuna di esse è in stato ârejectedâ, allora lâevento è generato.
Nellâesempio, il .catch aggiunto da setTimeout viene es, ovviamente lo fa, ma dopo, quando unhandledrejection è già avvenuto.
Se non fossimo a conoscenza della coda dei microtask, potremmo chiederci: âPerché il gestore di unhandledrejection viene eseguito? Abbiamo catturato lâerrore!â.
Ma ora sappiamo che unhandledrejection è generato quando la coda dei microtask è completa: il motore esamina le promise e, se una di esse è in stato ârejectedâ, allora lâevento viene innescato.
Nellâesempio sopra, anche il .catch aggiunto da setTimeout viene innescato, ma dopo, quando unhandledrejection è già avvenuto, quindi questo non cambia niente.
Riepilogo
La gestione delle promise è sempre asincrona, dato che tutte le azioni delle promise passano attraverso la coda âpromise jobsâ, anche chiamata âmicrotask queueâ (termine di v8).
Così, i gestori .then/catch/finally sono sempre chiamati dopo che il codice corrente è finito.
Se abbiamo bisogno della certezza che un pezzo di codice sia eseguito dopo .then/catch/finally, possiamo aggiungerlo ad una chiamata .then in catena.
Nella maggior parte dei motori JavaScript, inclusi i browser e Node.js, il concetto di microtask è strettamente legato al âloop degli eventâ (event loop) ed ai âmacrotasksâ. Dato che questi non hanno una relazione diretta con le promise, sono coperti in unâaltra parte del tutorial, nel capitolo Event loop: microtasks e macrotasks.
Commenti
<code>, per molte righe â includile nel tag<pre>, per più di 10 righe â utilizza una sandbox (plnkr, jsbin, codepenâ¦)