Uvod
U JavaScript-u postoji izraz za deo koda sa ugnježdenim callback funkcijama pod nazivom “callback hell” ili “pyramid of doom”. Debagovanje takvog programskog koda pa čak i samo razumevanje je veoma otežano. Sa ES2015 standardom je došla nova syntax-a pod nazivom “promise” (srp. obećanje), koja sa svojim API-jem obezbedjuje bolji i pregledniji način za organizovanje callback funkcija. Ovo se naručito primećuje u radu sa asinhronim operacijama jer sa promisima sintaksa veoma liči na standardnu sinhronu sintaksu.
Primer – “callback hell”
Najpoznatija i najčešća primena callback funkcija je “hendlovanje” akcija nakon izvršenog dogadjaja. Upravo je “hendlovanje” medjusobno zavisnih dogadjaja najčešće uzrok “callback hell-a”. U ovom primeru je prikazan izgled takvog koda:
1 2 3 4 5 6 7 8 9 |
callEndpoint("api/getidbyusername/nekoime", function (result) { callEndpoint("api/getifolowersbyid/" + result.userID, function (result) { callEndpoint("api/nekidrugizahtev/" result.folowers, function (result) { callEndpoint("api/nekidrugizahtev/" result.folowers, function (result) { // uhh ovde je već haos! }); }); }); }); |
Šta je Promise?
Promise je javascript objekat koji predstavlja “placeholder” za rezultate asihrone funkcije sve dok traje izvršavanje asinhrone operacije.
Promise je sinhrono vraćen objekat pri asinhronoj operaciji, koji predstavlja privremenu zamenu za moguće rezultate te asinhrone operacije.
Promise umesto krajnje vrednosti, daje obećanje da će dostaviti tu vrednost u nekom trenutku u budućnosti. Pošto su promise objekti privremena zamena za buduću eventualnu vrednost, to nam omogućava da preko njega zakačimo handler-e za budući rezultat asinhrone operacije. Sa ovom novom mogućnosti smo skoro izjednačili asinhrone operacije sa sinhronim. Sada i sinhrone i asinhrone operacije mogu da vraćaju neku vrednosti, stim što sinhrone odmah vraćaju krajnji podatak a asinhrone “placeholder” za budući podatak.
Asinhrona funkcija može imati dva moguća krajnja rezultata, a to su “uspešno izvršena operacija” ili “neuspešno izvršena opercija”, dok se promise može nalaziti u jednom od tri stanja:
- Pending – kada se asinhrona radnja još uvek izvršava
- Fulfill – kada je asinhrona radnja završena uspešno
- Reject – kada je asinhrona radnja neuspešno završena greškom
Ceo promise mehanizam se može podeliti na dva dela:
- Kreiranje promisa unutar asinhrone funkcije
- Korišćenje kreiranog promisa (kod se nalazi izvan asinhrone funkcije)
Kreiranje promisa
Da bi se budući krajnji rezultat asinhrone funkcije zamenio sa promise objektom, potrebno je da funkcija vrati novi promise objekat kroz svoj kod:
1 2 3 |
function asinhronaFunkcija() { return new Promise(function (){...}); } |
Svakom novom obećanju se kroz parametar prosledjuje funkcija (tzv. executor funkcija) koja obradjuje samu asinhronu operaciju i buduće rezultate te asinhrone operacije. Nama je interesantan deo gde obradjuje eventualne rezultate asinhrone operacije, kada u zavisnosti od uspešnosti operacije poziva jednu od dve funkcije koje su joj prosledjene kao parametri:
- Funkcija resolve() se poziva u delu koda koji obradjuje uspešno završenu asinhronu operaciju. Parametar ove funkcije predstavlja dobijeni podatak iz uspešno završene operacije, stoga se funkcija resolve() koristi da kroz svoj parametar prosledi rezultujući podatak odgovarajućoj “handler” metodi npr. then() ili Promise.all()…
- Funkcija “reject()” se poziva u delu koda koji obradjuje slučaj kada se pojavi problem sa izvršavanjem asinhrone operacije. Ona kroz svoj parametar prosledjuje razlog neuspešnosti asinhrone operacije odgovarajućem “hendleru”, najčešće catch() metodi.
1 2 3 4 5 6 7 8 9 10 11 12 13 |
function asinhronaFunkcija() { return new Promise(function (resolve, reject) { //...kod za asinhronu operaciju... if (uspešna operacija) { resolve(result_value); } else { reject(error); } } ); } |
Korišćenje promisa
Korišćenje promisa podrazumeva obradu eventualnih rezultata dobijenih po završetku asinhrone operacije.
Promise.prototype.then()
Promise reaguje na promenu svoga stanja pozivajući callback funkciju. Ukoliko je nakon promene stanja promise u stanju fullfiled poziva se metoda then(). Ova metoda prihvata dva parametra “onFulfilled” i “onRejected” koji su tipa funkcije.
1 |
Promise.prototype.then(onFulfilled(), onRejected()) |
Prva funkcija onFulfilled “hendluje” uspešno završenu asinhronu operaciju. Ona prihvata jedan parametar a kroz njega joj se prosledjuje podatak dobijen asinhronom operacijom. Dok druga funkcija “onRejected” “hendluje” neuspešno završenu asinhronu operaciju i takodje prihvata jedan parametar kroz koji joj se prosledjuje razlog neuspeha.
1 2 |
nekiPromise.then(function(podatak) { // deo koda kada je uspešna asinhrona operacija }, function(razlog) { // deo koda kada je neuspešna asinhrona operacija }); |
Primer
U ovom primeru kroz parametar funkcije resolve() prosledjujemo podatak (xhr.response) funkciji onFulfilled(), dok kroz parametar funkcije reject() prosledjujemo tip greške “new Error(xhr.statusText)” funkciji onRejected().
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
function makeRequest (method, url) { return new Promise(function (resolve, reject) { var xhr = new XMLHttpRequest(); xhr.open(method, url); xhr.onload = function() { if (xhr.status === 200) { resolve(xhr.response); } else { reject(new Error(xhr.statusText)); } }; xhr.onerror = () => reject(new Error("Network error")); xhr.send(); }); } // II deo - izvan asinhrone funkcije: makeRequest('GET', 'http://example.com') .then( function(data){console.log(data);}, function(err){console.error(err);} ) |
Promise.prototype.catch()
Ukoliko je nakon promene stanja, promise u stanju rejected, poziva se metoda catch().
1 |
Promise.prototype.catch(onRejected()) |
Metoda catch() kao parametar prihvata callback funkciju (za koju se koristi naziv “onRejected()”), koja je zadužena za prihvatanje i obradu greške.
OBJAŠNJENJE:
Iako metoda then() može da “hendluje” pored uspešnih rezultata i neuspešne, to se ne koristi tako često. Najčešće se metoda then() koristi da “hendluje” samo uspešan rezultat (koristi se samo prvi parametar), dok se za slučaj neuspešne operacije koristi catch() metoda.
1 |
asinhronaFunkcija().then(result_value => { ··· }).catch(error => { ··· }); |
Ili malo preglednije napisano:
1 2 3 |
asinhronaFunkcija() .then(result => { ··· }) .catch(error => { ··· }); |
Primer
U primeru je prikazano hendlovanje ajax asinhrone operacije sa promisom koji koristi then() i catch() metodu ali i sa “zastarelim” callback funkcijama:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 |
function makeRequest (method, url) { return new Promise(function (resolve, reject) { var xhr = new XMLHttpRequest(); xhr.open(method, url); xhr.onload = function() { if (xhr.status === 200) { resolve(xhr.response); } else { reject(new Error(xhr.statusText)); } }; xhr.onerror = () => reject(new Error("Network error")); xhr.send(); }); } // II deo - izvan asinhrone funkcije: makeRequest('GET', 'http://example.com') .then(function (data) { console.log(data); }) .catch(function (err) { console.error('Uhh imamo problem!', err); }); |
U ovom primeru je preko parametra funkcije resolve() prosledjen “xhr.responseText” u “data” parametar then() metode, a preko parametra funkcije reject() je prosledjen “new Error(“Network error”)” u “err”. parametar catch() metode
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
function makeRequest (method, url, callbackFunkcija) { var xhr = new XMLHttpRequest(); xhr.open(method, url); xhr.onload = function () { callbackFunkcija(null, xhr.response); // null se prosledjuje umesto erorr-a }; xhr.onerror = function () { callbackFunkcija(error); }; xhr.send(); } makeRequest('GET', 'http://example.com', function (error, data) { if (error) { throw error; } console.log(data); }); |
Ulančavanje promisa
Pošto metode then() i catch(), uvek vraćaju novi promise, one se mogu ulančavati. Koristeći ovu mogućnost se otvaraju vrata da se na elegantan nači reši problem sa mnogo ulančanih dogadjaja koje pozivaju callback funkcije, poznatiji kao “callback hell”.
Primer
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 |
const promise = new Promise( function(resolve, reject) { setTimeout( () => { resolve( [ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9] ); }, 2000); }); function ispisiRezultat(value) { console.log(value); return value; } function samoParni(array) { return array.filter( (value) => { return (value % 2) === 0; }); } function sumaClanovaNiza(array) { return array.reduce( (a, b) => { return a + b; }, 0); } function errorHandler(err) { console.log("ERROR"); console.log(err); } promise .then(ispisiRezultat) // Vraca: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9] .then(samoParni) .then(ispisiRezultat) // Vraca: [0, 2, 4, 6, 8] .then(sumaClanovaNiza) .then(ispisiRezultat) // Vraca: 20 .catch(errorHandler); |
Promise.resolve()
Promise.resolve(value) je metoda koja pravi promise od drugačijih tipova. Može da primi tri različita tipa kao parametar, nakon čega vraća promise:
- Promise.resolve(vrednost)
Kada se ovoj metodi prosledi neka vrednost, ta vrednost će biti prosledjena then() funkciji kroz njen parametar. - Promise.resolve(drugiPromise)
Kada se prosledi neki drugi promise tada se kroz parametar metode then() prosledjuje eventualno stanje prihvaćenog promise objekta. Ovo je najčešći slučaj, jer se koristi za konvertovanje promisa kreiranih od strane drugih biblioteka.
Primer
12345678910var pocetniPromise = Promise.resolve(33);var drugiPromise = Promise.resolve(pocetniPromise);drugiPromise.then(function(value) {console.log('value: ' + value);});console.log('pocetniPromise === drugiPromise je ' + (pocetniPromise === drugiPromise));// Izlaz na konzoli (obrati pažnju na redosled ispisa!):// "pocetniPromise === drugiPromise je true"// value: 33Obratiti pažnju na redosled štampanja u konzoli, redosled je drugačiji nego što je u kodu, zato što je “then handlers” pozvan asinhrono. Više o ovome pogledajte u članku Osnove asinhronog programiranja u JavaScript-u
- Promise.resolve(thenableObject)
Ova metoda može da prihvati kao parametar i tzv. “thenable object”:
Primer
12345678910var p1 = Promise.resolve({then: function(onFulfill, onReject) { onFulfill('fulfilled!'); }});console.log(p1 instanceof Promise) // true, object casted to a Promisep1.then(function(v) {console.log(v); // "fulfilled!"}, function(e) {// not called});
Promise.reject()
Promise.reject(reason) uzima za “reason object” String ili Error, a vraća promise koji se nalazi u stanju rejected uz odgovarajući razlog odbijanja.
1 2 |
Promise.reject(new Error('fail')) .then(function() {ovaj deo je za uspešno i ne poziva se}, function(error) { console.log(error); }); |
Promise.all()
Promise.all(iterable) kao parametar uzima iterabilnu listu promise objekata. Promise.all() metod vraća jedan promise u trenutku kada su svi promisi sa liste uspešno rešeni. Promise.all() metoda će vratiti “promis”, čak i kada joj se prosledi prazna lista ili element koji nije promise (pogledaj primer). Kod ove metode redosled izvršavanja liste promisa nije zagarantovan, jedino je zagarantovano da će vratiti krajnji “promis”, kada svi promisi sa liste budu fullfiled.
Primer
1 2 3 4 5 6 7 8 9 |
var p1 = Promise.resolve(3); var p2 = 1337; var p3 = new Promise((resolve, reject) => { setTimeout(resolve, 100, 'foo'); }); Promise.all([p1, p2, p3]).then(values => { console.log(values); // Vraća: [3, 1337, "foo"] }); |
U slučaju da je jedan promise iz liste u stanju rejected, metoda Promise.all() vraća odbijen promise bez obzira da li u listi postoji neki promis koji je uspešan. Uz odbijen promis se vraća i razlog odbijanja. U slučaju da ima više odbijenih promisa, prosledjeni razlog je od prvog odbijenog promisa!
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 |
var p1 = new Promise((resolve, reject) => { setTimeout(resolve, 1000, 'one'); }); var p2 = new Promise((resolve, reject) => { setTimeout(resolve, 2000, 'two'); }); var p3 = new Promise((resolve, reject) => { setTimeout(resolve, 3000, 'three'); }); var p4 = new Promise((resolve, reject) => { reject('Ovaj promise je namerno odbijen'); }); var p5 = new Promise((resolve, reject) => { setTimeout(resolve, 4000, 'five'); }); Promise.all([p1, p2, p3, p4, p5]).then(values => { console.log(values); }).catch(reason => { console.log("Nama svih sastojaka", reason); }); // Vraća: "Ovaj promise je namerno odbijen" |
Asinhronost i sinhronost Promise.all()
Promise.all() se u svim slučajevima ponaša asinhrono osim u slučaju kada je umesto liste prosledjen prazan objekt, tada se ponaša sinhrono.
Primer
1 2 3 4 5 6 7 8 9 10 11 12 |
var prazanPromise = Promise.all([]); var listaPromise = [Promise.resolve("neki podatak"), Promise.resolve(123)]; var nekiPromise = Promise.all(listaPromise); // Sinhrono stampanje izlaza console.log(prazanPromise) // Vraća: Promise { <state>: "fulfilled", <value>: Array[0] } console.log(nekiPromise); // Vraća: Promise { <state>: "pending", <value>: undefined } // Asinhrono stampanje izlaza setTimeout(function(){ console.log(nekiPromise); // Vraća: Promise { <state>: "fulfilled", <value>: Array[2] } }); |
Obratite pažnju da je pri direktnom sinhronom stampanju “prazanPromise” u stanju fullfiled, dok je standardni promise pod nazivom “nekiPromise” u stanju pending.
Kada koristiti “Promise.all()” a kada “Promise.prototype.then()”?
Izbor se svodi na to da li je bitan redosled izvršavanja promisa ili ne. U sledećem primeru je bitan redosled izvršavanja promisa pa se koristi Promise.prototype.then():
1 2 3 |
nabaviDrvo() .then(() => napraviCamac()) .then(ploviRekom()); |
Dok u narednom primeru za pravljenje betona nije bitan redosled kada će koji promise biti završen, već je samo bitan trenutak kada su svi spremni, pa je preporuka da se koristi Promise.all():
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
var nabaviSljunak = new Promise((resolve, reject) => { setTimeout(resolve, 1000, 'one'); }); var nabaviCement = new Promise((resolve, reject) => { setTimeout(resolve, 2000, 'two'); }); var nabaviAditiv = new Promise((resolve, reject) => { setTimeout(resolve, 3000, 'three'); }); Promise.all([nabaviSljunak, nabaviCement, nabaviAditiv]) .then(values => { //napravi beton }) .catch(reason => { console.log(reason) }); |
Promise.race()
Promise.race(iterable) je metoda koja takodje kao parametar uzima iterabilnu listu promise objekata a vraća obećanje. Ova metoda vraća rezultat koji “stigne prvi”, nebitno da li je uspešan ili ne. Pa tako može biti vraćen promise ili sa krajnjim podatkom kod uspešno izvršene asinhrone operacije ili sa razlogom za neuspešnu operaciju.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 |
var p1 = new Promise(function(resolve, reject) { setTimeout(resolve, 500, "one"); }); var p2 = new Promise(function(resolve, reject) { setTimeout(resolve, 100, "two"); }); Promise.race([p1, p2]).then(function(value) { console.log(value); // "two" // Both resolve, but p2 is faster }); //----------------------------------------------------- var p3 = new Promise(function(resolve, reject) { setTimeout(resolve, 100, "three"); }); var p4 = new Promise(function(resolve, reject) { setTimeout(reject, 500, "four"); }); Promise.race([p3, p4]).then(function(value) { console.log(value); // "three" // p3 is faster, so it resolves }, function(reason) { // Not called }); //----------------------------------------------------- var p5 = new Promise(function(resolve, reject) { setTimeout(resolve, 500, "five"); }); var p6 = new Promise(function(resolve, reject) { setTimeout(reject, 100, "six"); }); Promise.race([p5, p6]).then(function(value) { // Not called }, function(reason) { console.log(reason); // "six" // p6 is faster, so it rejects }); |
Svaka čast na objašnjavanju i trudu. Tek kad neko pokuša da obajsni postaje uočljivo se koliki idiot je smišljao ovaj programski jezik i koliko je ovo zavitlavanje ljudi.
Indeed, bro!
Slažem se. Iz dana u dan ovaj programski jezik postaje sve vise natrpan kojekakvim zajebancijama. Često se upitam šta mi sve ovo treba.
Hvala na podršci!
Svaka cast na objasnjavanju, mnogo sam naucio sa ovoga sjata i dalje ucim. 🙂
Nadam se da ne planirate stati sa pisanjem 😀