Jak zwrócić odpowiedź z wywołania asynchronicznego?
Mam funkcję, foo
która wysyła asynchroniczne żądanie. Jak mogę zwrócić odpowiedź / wynik z foo
?
Próbowałem zwrócić wartość z wywołania zwrotnego, a także przypisać wynik do zmiennej lokalnej wewnątrz funkcji i zwrócić tę, ale żaden z tych sposobów nie zwraca odpowiedzi (wszystkie zwracają undefined
lub jakąkolwiek początkową wartość zmiennej result
) .
Przykład użycia ajax
funkcji jQuery :
function foo() {
var result;
$.ajax({
url: '...',
success: function(response) {
result = response;
// return response; // <- I tried that one as well
}
});
return result; // It always returns `undefined`
}
Przykład z użyciem node.js:
function foo() {
var result;
fs.readFile("path/to/file", function(err, data) {
result = data;
// return data; // <- I tried that one as well
});
return result; // It always returns `undefined`
}
Przykład użycia then
bloku obietnicy:
function foo() {
var result;
fetch(url).then(function(response) {
result = response;
// return response; // <- I tried that one as well
});
return result; // It always returns `undefined`
}
Odpowiedzi
→ Aby uzyskać bardziej ogólne wyjaśnienie zachowania asynchronicznego z różnymi przykładami, zobacz Dlaczego moja zmienna pozostaje niezmieniona po zmodyfikowaniu jej wewnątrz funkcji? - Odwołanie do kodu asynchronicznego
→ Jeśli już rozumiesz problem, przejdź do poniższych możliwych rozwiązań.
Problem
W Ajax oznacza asynchroniczny . Oznacza to, że wysłanie żądania (a raczej otrzymanie odpowiedzi) jest usuwane z normalnego przepływu wykonywania. W naszym przykładzie zwraca natychmiast, a następna instrukcja ,, jest wykonywana przed wywołaniem funkcji, którą przekazałeś jako callback.$.ajax
return result;
success
Oto analogia, która, miejmy nadzieję, wyjaśnia różnicę między przepływem synchronicznym i asynchronicznym:
Synchroniczny
Wyobraź sobie, że dzwonisz do znajomego i prosisz go, aby wyszukał coś dla Ciebie. Chociaż może to chwilę potrwać, czekasz na telefon i gapisz się w przestrzeń, aż znajomy udzieli Ci odpowiedzi, której potrzebujesz.
To samo dzieje się, gdy wywołujesz funkcję zawierającą „normalny” kod:
function findItem() {
var item;
while(item_not_found) {
// search
}
return item;
}
var item = findItem();
// Do something with item
doSomethingElse();
Nawet jeśli wykonanie findItem
może zająć dużo czasu, każdy kod nadchodzący później var item = findItem();
musi poczekać, aż funkcja zwróci wynik.
Asynchroniczny
Z tego samego powodu dzwonisz ponownie do przyjaciela. Ale tym razem mówisz mu, że się spieszysz i powinien oddzwonić na Twój telefon komórkowy. Rozłączasz się, wychodzisz z domu i robisz wszystko, co zaplanowałeś. Gdy przyjaciel oddzwoni, masz do czynienia z informacjami, które ci przekazał.
Dokładnie to się dzieje, gdy wykonujesz żądanie Ajax.
findItem(function(item) {
// Do something with the item
});
doSomethingElse();
Zamiast czekać na odpowiedź, wykonywanie jest kontynuowane natychmiast i wykonywana jest instrukcja po wywołaniu Ajax. Aby w końcu uzyskać odpowiedź, podajesz funkcję, która zostanie wywołana po otrzymaniu odpowiedzi, wywołanie zwrotne (zauważ coś? Oddzwoń ?). Każda instrukcja występująca po tym wywołaniu jest wykonywana przed wywołaniem funkcji zwrotnej.
Rozwiązania)
Przyjmij asynchroniczną naturę JavaScript! Chociaż niektóre operacje asynchroniczne zapewniają odpowiedniki synchroniczne (podobnie jak „Ajax”), generalnie odradza się ich używanie, szczególnie w kontekście przeglądarki.
Dlaczego jest źle, pytasz?
JavaScript działa w wątku interfejsu użytkownika przeglądarki, a każdy długotrwały proces zablokuje interfejs użytkownika, przez co przestanie on odpowiadać. Dodatkowo istnieje górny limit czasu wykonywania JavaScript, a przeglądarka zapyta użytkownika, czy kontynuować wykonywanie, czy nie.
Wszystko to jest naprawdę złe dla użytkownika. Użytkownik nie będzie w stanie stwierdzić, czy wszystko działa dobrze, czy nie. Ponadto efekt będzie gorszy w przypadku użytkowników z wolnym połączeniem.
Poniżej przyjrzymy się trzem różnym rozwiązaniom, które budują się jeden na drugim:
- Obietnice z
async/await
(ES2017 +, dostępne w starszych przeglądarkach, jeśli używasz transpilera lub regeneratora) - Wywołania zwrotne (popularne w węźle)
- Obietnice z
then()
(ES2015 +, dostępne w starszych przeglądarkach, jeśli używasz jednej z wielu bibliotek obietnic)
Wszystkie trzy są dostępne w obecnych przeglądarkach oraz w węźle 7+.
ES2017 +: Obietnice dotyczące async/await
Wersja ECMAScript wydana w 2017 roku wprowadziła obsługę funkcji asynchronicznych na poziomie składni . Za pomocą async
i await
możesz pisać asynchronicznie w „stylu synchronicznym”. Kod jest nadal asynchroniczny, ale jest łatwiejszy do odczytania / zrozumienia.
async/await
opiera się na obietnicach: async
funkcja zawsze zwraca obietnicę. await
„rozpina” obietnicę i albo skutkuje wartością, z jaką obietnica została rozwiązana, albo zgłasza błąd, jeśli obietnica została odrzucona.
Ważne: możesz używać tylko await
wewnątrz async
funkcji. Obecnie najwyższy poziom await
nie jest jeszcze obsługiwany, więc może być konieczne utworzenie asynchronicznego IIFE ( natychmiastowo wywoływanego wyrażenia funkcji ), aby rozpocząć async
kontekst.
Możesz przeczytać więcej o asynci awaitna MDN.
Oto przykład, który opiera się na powyższym opóźnieniu:
// Using 'superagent' which will return a promise.
var superagent = require('superagent')
// This is isn't declared as `async` because it already returns a promise
function delay() {
// `delay` returns a promise
return new Promise(function(resolve, reject) {
// Only `delay` is able to resolve or reject the promise
setTimeout(function() {
resolve(42); // After 3 seconds, resolve the promise with value 42
}, 3000);
});
}
async function getAllBooks() {
try {
// GET a list of book IDs of the current user
var bookIDs = await superagent.get('/user/books');
// wait for 3 seconds (just for the sake of this example)
await delay();
// GET information about each book
return await superagent.get('/books/ids='+JSON.stringify(bookIDs));
} catch(error) {
// If any of the awaited promises was rejected, this catch block
// would catch the rejection reason
return null;
}
}
// Start an IIFE to use `await` at the top level
(async function(){
let books = await getAllBooks();
console.log(books);
})();
Obsługiwane są aktualne wersje przeglądarek i węzłówasync/await
. Możesz również wspierać starsze środowiska, przekształcając swój kod do ES5 za pomocą regeneratora (lub narzędzi, które używają regeneratora, takich jak Babel ).
Pozwól funkcjom akceptować wywołania zwrotne
Wywołanie zwrotne ma miejsce, gdy funkcja 1 jest przekazywana do funkcji 2. Funkcja 2 może wywołać funkcję 1, gdy jest gotowa. W kontekście procesu asynchronicznego wywołanie zwrotne zostanie wywołane za każdym razem, gdy proces asynchroniczny zostanie zakończony. Zwykle wynik jest przekazywany do wywołania zwrotnego.
W przykładzie pytania możesz foo
zaakceptować oddzwonienie i użyć go jako success
oddzwonienia. Więc to
var result = foo();
// Code that depends on 'result'
staje się
foo(function(result) {
// Code that depends on 'result'
});
Tutaj zdefiniowaliśmy funkcję „inline”, ale możesz przekazać dowolne odwołanie do funkcji:
function myCallback(result) {
// Code that depends on 'result'
}
foo(myCallback);
foo
sama jest zdefiniowana w następujący sposób:
function foo(callback) {
$.ajax({
// ...
success: callback
});
}
callback
będzie odnosić się do funkcji, do której przekazujemy, foo
gdy ją wywołujemy, i do której ją przekazujemy success
. To znaczy, gdy żądanie Ajax się powiedzie, $.ajax
wywoła callback
i przekaże odpowiedź do wywołania zwrotnego (do którego można się odwołać result
, ponieważ tak zdefiniowaliśmy wywołanie zwrotne).
Możesz również przetworzyć odpowiedź przed przekazaniem jej do wywołania zwrotnego:
function foo(callback) {
$.ajax({
// ...
success: function(response) {
// For example, filter the response
callback(filtered_response);
}
});
}
Pisanie kodu za pomocą wywołań zwrotnych jest łatwiejsze niż mogłoby się wydawać. W końcu JavaScript w przeglądarce jest w dużej mierze zależny od zdarzeń (zdarzenia DOM). Otrzymanie odpowiedzi Ajax to nic innego jak wydarzenie.
Trudności mogą się pojawić, gdy będziesz musiał pracować z kodem innych firm, ale większość problemów można rozwiązać, po prostu przemyślając przepływ aplikacji.
ES2015 +: Obietnice z then ()
Obietnica API jest nowa funkcja ECMAScript 6 (ES2015), ale ma dobrą obsługę przeglądarki już. Istnieje również wiele bibliotek, które implementują standardowe API Promises i udostępniają dodatkowe metody ułatwiające użycie i kompozycję funkcji asynchronicznych (np. Bluebird ).
Obietnice są pojemnikami na przyszłe wartości. Gdy obietnica otrzyma wartość (zostanie rozwiązana ) lub gdy zostanie anulowana ( odrzucona ), powiadamia wszystkich swoich „słuchaczy”, którzy chcą uzyskać dostęp do tej wartości.
Zaletą w stosunku do zwykłych wywołań zwrotnych jest to, że pozwalają one oddzielić kod i są łatwiejsze w komponowaniu.
Oto przykład wykorzystania obietnicy:
function delay() {
// `delay` returns a promise
return new Promise(function(resolve, reject) {
// Only `delay` is able to resolve or reject the promise
setTimeout(function() {
resolve(42); // After 3 seconds, resolve the promise with value 42
}, 3000);
});
}
delay()
.then(function(v) { // `delay` returns a promise
console.log(v); // Log the value once it is resolved
})
.catch(function(v) {
// Or do something else if it is rejected
// (it would not happen in this example, since `reject` is not called).
});
W przypadku naszego wywołania Ajax moglibyśmy użyć takich obietnic:
function ajax(url) {
return new Promise(function(resolve, reject) {
var xhr = new XMLHttpRequest();
xhr.onload = function() {
resolve(this.responseText);
};
xhr.onerror = reject;
xhr.open('GET', url);
xhr.send();
});
}
ajax("/echo/json")
.then(function(result) {
// Code depending on result
})
.catch(function() {
// An error occurred
});
Opisanie wszystkich zalet, które oferuje obietnica, wykracza poza zakres tej odpowiedzi, ale jeśli piszesz nowy kod, powinieneś poważnie się nad nimi zastanowić. Zapewniają doskonałą abstrakcję i separację twojego kodu.
Więcej informacji o obietnicach: HTML5 rządzi - JavaScript Promises
Uwaga dodatkowa: odroczone obiekty jQuery
Obiekty odroczone to niestandardowa implementacja obietnic jQuery (przed standaryzacją Promise API). Zachowują się prawie jak obietnice, ale udostępniają nieco inne API.
Każda metoda Ajax jQuery zwraca już "odroczony obiekt" (właściwie obietnicę odroczonego obiektu), który możesz po prostu zwrócić z funkcji:
function ajax() {
return $.ajax(...);
}
ajax().done(function(result) {
// Code depending on result
}).fail(function() {
// An error occurred
});
Uwaga dodatkowa: obietnice
Należy pamiętać, że obietnice i odroczone obiekty są tylko kontenerami na przyszłą wartość, a nie samą wartością. Na przykład załóżmy, że masz:
function checkPassword() {
return $.ajax({
url: '/password',
data: {
username: $('#username').val(), password: $('#password').val()
},
type: 'POST',
dataType: 'json'
});
}
if (checkPassword()) {
// Tell the user they're logged in
}
Ten kod źle rozumie powyższe problemy z asynchronią. W szczególności $.ajax()
nie zamraża kodu, gdy sprawdza stronę „/ hasło” na serwerze - wysyła żądanie do serwera i podczas oczekiwania natychmiast zwraca obiekt jQuery Ajax Deferred, a nie odpowiedź z serwera. Oznacza to, że if
instrukcja zawsze pobierze ten obiekt odroczony, potraktuje go jako true
i będzie postępować tak, jakby użytkownik był zalogowany. Niedobrze.
Ale poprawka jest łatwa:
checkPassword()
.done(function(r) {
if (r) {
// Tell the user they're logged in
} else {
// Tell the user their password was bad
}
})
.fail(function(x) {
// Tell the user something bad happened
});
Niezalecane: synchroniczne połączenia „Ajax”
Jak wspomniałem, niektóre (!) Operacje asynchroniczne mają odpowiedniki synchroniczne. Nie zalecam ich używania, ale dla zachowania kompletności, oto jak wykonać wywołanie synchroniczne:
Bez jQuery
Jeśli bezpośrednio używasz XMLHttpRequestobiektu, przekaż false
jako trzeci argument do .open.
jQuery
Jeśli używasz jQuery , możesz ustawić async
opcję na false
. Zauważ, że ta opcja jest przestarzała od wersji jQuery 1.8. Następnie możesz nadal użyć success
wywołania zwrotnego lub uzyskać dostęp do responseText
właściwości obiektu jqXHR :
function foo() {
var jqXHR = $.ajax({
//...
async: false
});
return jqXHR.responseText;
}
Jeśli używasz innej metody jQuery AJAX, takich jak $.get
, $.getJSON
itp, trzeba go zmienić $.ajax
(ponieważ można przechodzić tylko do parametrów konfiguracyjnych $.ajax
).
Heads-up! Nie jest możliwe wykonanie synchronicznego żądania JSONP . JSONP z natury jest zawsze asynchroniczny (jeszcze jeden powód, aby nawet nie brać pod uwagę tej opcji).
Jeśli nie używasz jQuery w swoim kodzie, ta odpowiedź jest dla Ciebie
Twój kod powinien być podobny do tego:
function foo() {
var httpRequest = new XMLHttpRequest();
httpRequest.open('GET', "/echo/json");
httpRequest.send();
return httpRequest.responseText;
}
var result = foo(); // always ends up being 'undefined'
Felix Kling wykonał świetną robotę, pisząc odpowiedź dla osób używających jQuery dla AJAX. Postanowiłem zapewnić alternatywę dla osób, które tego nie robią.
( Uwaga, dla osób korzystających z nowego fetchAPI, Angular lub obietnic dodałem kolejną odpowiedź poniżej )
Przed czym stoisz
To jest krótkie podsumowanie „Wyjaśnienia problemu” z innej odpowiedzi, jeśli po przeczytaniu tego nie masz pewności, przeczytaj to.
W AJAX oznacza asynchroniczny . Oznacza to, że wysłanie żądania (a raczej otrzymanie odpowiedzi) jest usuwane z normalnego przepływu wykonywania. W naszym przykładzie zwraca natychmiast, a następna instrukcja ,, jest wykonywana przed wywołaniem funkcji, którą przekazałeś jako callback..sendreturn result;
success
Oznacza to, że kiedy zwracasz, zdefiniowany detektor nie został jeszcze wykonany, co oznacza, że zwracana wartość nie została zdefiniowana.
Oto prosta analogia
function getFive(){
var a;
setTimeout(function(){
a=5;
},10);
return a;
}
(Skrzypce)
Wartość a
zwracana jest, undefined
ponieważ a=5
część nie została jeszcze wykonana. AJAX działa w ten sposób, zwracasz wartość, zanim serwer będzie miał szansę powiedzieć przeglądarce, jaka jest ta wartość.
Jednym z możliwych rozwiązań tego problemu jest ponowne aktywowanie kodu , informując program, co ma zrobić po zakończeniu obliczeń.
function onComplete(a){ // When the code completes, do this
alert(a);
}
function getFive(whenDone){
var a;
setTimeout(function(){
a=5;
whenDone(a);
},10);
}
Nazywa się to CPS . Zasadniczo przekazujemy getFive
akcję do wykonania po jej zakończeniu, mówimy naszemu kodowi, jak ma zareagować po zakończeniu zdarzenia (jak nasze wywołanie AJAX lub w tym przypadku przekroczenie limitu czasu).
Sposób użycia będzie:
getFive(onComplete);
Co powinno ostrzec „5” na ekranie. (Skrzypce) .
Możliwe rozwiązania
Zasadniczo istnieją dwa sposoby rozwiązania tego problemu:
- Spraw, aby połączenie AJAX było synchroniczne (nazwijmy to SJAX).
- Przebuduj swój kod, aby działał poprawnie z wywołaniami zwrotnymi.
1. Synchroniczny AJAX - nie rób tego !!
Jeśli chodzi o synchroniczny AJAX, nie rób tego! Odpowiedź Felixa rodzi przekonujące argumenty na temat tego, dlaczego to zły pomysł. Podsumowując, zamrozi przeglądarkę użytkownika, dopóki serwer nie zwróci odpowiedzi i spowoduje bardzo złe wrażenia użytkownika. Oto kolejne krótkie podsumowanie zaczerpnięte z MDN na temat tego, dlaczego:
XMLHttpRequest obsługuje komunikację synchroniczną i asynchroniczną. Ogólnie jednak ze względu na wydajność żądania asynchroniczne powinny być preferowane względem żądań synchronicznych.
Krótko mówiąc, żądania synchroniczne blokują wykonanie kodu ... ... może to spowodować poważne problemy ...
Jeśli mają to zrobić, można przekazać flagi: Oto jak to zrobić:
var request = new XMLHttpRequest();
request.open('GET', 'yourURL', false); // `false` makes the request synchronous
request.send(null);
if (request.status === 200) {// That's HTTP for 'ok'
console.log(request.responseText);
}
2. Restrukturyzacja kodu
Pozwól swojej funkcji zaakceptować wywołanie zwrotne. W tym przykładzie kod foo
może przyjąć oddzwonienie. Powiemy naszemu kodowi, jak ma reagować po foo
zakończeniu.
Więc:
var result = foo();
// code that depends on `result` goes here
Staje się:
foo(function(result) {
// code that depends on `result`
});
Tutaj przekazaliśmy anonimową funkcję, ale równie łatwo moglibyśmy przekazać odniesienie do istniejącej funkcji, nadając jej wygląd:
function myHandler(result) {
// code that depends on `result`
}
foo(myHandler);
Aby uzyskać więcej informacji na temat tego rodzaju projektowania wywołań zwrotnych, sprawdź odpowiedź Felixa.
Teraz zdefiniujmy foo, aby działało odpowiednio
function foo(callback) {
var httpRequest = new XMLHttpRequest();
httpRequest.onload = function(){ // when the request is loaded
callback(httpRequest.responseText);// we're calling our method
};
httpRequest.open('GET', "/echo/json");
httpRequest.send();
}
(skrzypce)
Teraz sprawiliśmy, że nasza funkcja foo akceptuje akcję do wykonania, gdy AJAX zakończy się pomyślnie, możemy to jeszcze bardziej rozszerzyć, sprawdzając, czy stan odpowiedzi jest inny niż 200 i działając odpowiednio (utwórz procedurę obsługi błędów itp.) Skutecznie rozwiązujemy nasz problem.
Jeśli nadal masz trudności ze zrozumieniem tego, przeczytaj przewodnik wprowadzający do technologii AJAX w MDN.
XMLHttpRequest 2 (przede wszystkim przeczytaj odpowiedzi Benjamina Gruenbauma i Felixa Klinga )
Jeśli nie używasz jQuery i chcesz ładnego, krótkiego XMLHttpRequest 2, który działa na nowoczesnych przeglądarkach, a także w przeglądarkach mobilnych, sugeruję użycie go w ten sposób:
function ajax(a, b, c){ // URL, callback, just a placeholder
c = new XMLHttpRequest;
c.open('GET', a);
c.onload = b;
c.send()
}
Jak widzisz:
- Jest krótszy niż wszystkie inne wymienione funkcje.
- Wywołanie zwrotne jest ustawiane bezpośrednio (więc nie ma dodatkowych niepotrzebnych zamknięć).
- Używa nowego onload (więc nie musisz sprawdzać stanu readystate &&)
- Jest kilka innych sytuacji, których nie pamiętam, a które powodują, że XMLHttpRequest 1 jest denerwujący.
Istnieją dwa sposoby uzyskania odpowiedzi na to wywołanie Ajax (trzy przy użyciu nazwy zmiennej XMLHttpRequest):
Najprostszy:
this.response
Lub jeśli z jakiegoś powodu bind()
oddzwonisz do klasy:
e.target.response
Przykład:
function callback(e){
console.log(this.response);
}
ajax('URL', callback);
Lub (powyższy jest lepszy, anonimowe funkcje zawsze stanowią problem):
ajax('URL', function(e){console.log(this.response)});
Nic prostszego.
Teraz niektórzy zapewne powiedzą, że lepiej jest użyć nazwy zmiennej onreadystatechange lub nawet nazwy zmiennej XMLHttpRequest. To jest źle.
Sprawdź zaawansowane funkcje XMLHttpRequest
Obsługuje wszystkie * nowoczesne przeglądarki. Mogę potwierdzić, ponieważ używam tego podejścia, ponieważ istnieje XMLHttpRequest 2. Nigdy nie miałem żadnych problemów ze wszystkimi przeglądarkami, których używam.
onreadystatechange jest przydatne tylko wtedy, gdy chcesz uzyskać nagłówki w stanie 2.
Używanie XMLHttpRequest
nazwy zmiennej jest kolejnym dużym błędem, ponieważ musisz wykonać wywołanie zwrotne wewnątrz zamknięć onload / oreadystatechange, w przeciwnym razie je zgubiłeś.
Teraz, jeśli chcesz czegoś bardziej złożonego przy użyciu post i FormData, możesz łatwo rozszerzyć tę funkcję:
function x(a, b, e, d, c){ // URL, callback, method, formdata or {key:val},placeholder
c = new XMLHttpRequest;
c.open(e||'get', a);
c.onload = b;
c.send(d||null)
}
Ponownie ... to bardzo krótka funkcja, ale pobiera i publikuje.
Przykłady użycia:
x(url, callback); // By default it's get so no need to set
x(url, callback, 'post', {'key': 'val'}); // No need to set post data
Lub przekaż pełny element formularza ( document.getElementsByTagName('form')[0]
):
var fd = new FormData(form);
x(url, callback, 'post', fd);
Lub ustaw wartości niestandardowe:
var fd = new FormData();
fd.append('key', 'val')
x(url, callback, 'post', fd);
Jak widać, nie zaimplementowałem synchronizacji ... to zła rzecz.
Powiedziawszy to ... dlaczego nie zrobić tego w łatwy sposób?
Jak wspomniano w komentarzu, użycie error && synchronous całkowicie przełamuje sens odpowiedzi. Jaki jest dobry, krótki sposób na prawidłowe używanie Ajax?
Obsługa błędów
function x(a, b, e, d, c){ // URL, callback, method, formdata or {key:val}, placeholder
c = new XMLHttpRequest;
c.open(e||'get', a);
c.onload = b;
c.onerror = error;
c.send(d||null)
}
function error(e){
console.log('--Error--', this.type);
console.log('this: ', this);
console.log('Event: ', e)
}
function displayAjax(e){
console.log(e, this);
}
x('WRONGURL', displayAjax);
W powyższym skrypcie masz moduł obsługi błędów, który jest zdefiniowany statycznie, więc nie zagraża funkcji. Moduł obsługi błędów może być również używany do innych funkcji.
Jednak jedynym sposobem na usunięcie błędu jest napisanie nieprawidłowego adresu URL. W takim przypadku każda przeglądarka zgłasza błąd.
Programy obsługi błędów mogą być przydatne, jeśli ustawisz niestandardowe nagłówki, ustawisz responseType na bufor tablicy blob lub cokolwiek ...
Nawet jeśli podasz „POSTAPAPAP” jako metodę, nie zgłosi ona błędu.
Nawet jeśli przekażesz „fdggdgilfdghfldj” jako formdata, nie zgłosi to błędu.
W pierwszym przypadku błąd jest wewnątrz displayAjax()
pod this.statusText
as Method not Allowed
.
W drugim przypadku po prostu działa. Musisz sprawdzić po stronie serwera, czy przekazałeś prawidłowe dane postu.
cross-domain not allowed automatycznie zgłasza błąd.
W odpowiedzi błędu nie ma kodów błędów.
Jest tylko to, this.type
co jest ustawione na błąd.
Po co dodawać moduł obsługi błędów, jeśli nie masz całkowitej kontroli nad błędami? Większość błędów jest zwracana wewnątrz tego w funkcji zwrotnej displayAjax()
.
Tak więc: nie ma potrzeby sprawdzania błędów, jeśli możesz poprawnie skopiować i wkleić adres URL. ;)
PS: Jako pierwszy test napisałem x ('x', displayAjax) ... i otrzymałem całkowitą odpowiedź ... ??? Więc sprawdziłem folder, w którym znajduje się HTML, i był tam plik o nazwie „x.xml”. Więc nawet jeśli zapomnisz rozszerzenia swojego pliku XMLHttpRequest 2 ZNAJDĘ JĄ . LOL'd
Czytaj plik synchroniczny
Nie rób tego.
Jeśli chcesz na chwilę zablokować przeglądarkę, załaduj ładny, duży .txt
plik synchroniczny.
function omg(a, c){ // URL
c = new XMLHttpRequest;
c.open('GET', a, true);
c.send();
return c; // Or c.response
}
Teraz możesz to zrobić
var res = omg('thisIsGonnaBlockThePage.txt');
Nie ma innego sposobu, aby to zrobić w sposób nieasynchroniczny. (Tak, z pętlą setTimeout ... ale poważnie?)
Inną kwestią jest ... jeśli pracujesz z interfejsami API lub tylko plikami własnej listy lub cokolwiek innego, zawsze używasz różnych funkcji dla każdego żądania ...
Tylko jeśli masz stronę, na której ładujesz zawsze ten sam XML / JSON lub cokolwiek potrzebujesz tylko jednej funkcji. W takim przypadku zmodyfikuj trochę funkcję Ajax i zastąp b swoją funkcją specjalną.
Powyższe funkcje służą do podstawowego użytku.
Jeśli chcesz PRZEDŁUŻYĆ funkcję ...
Tak, możesz.
Używam wielu interfejsów API, a jedną z pierwszych funkcji, które integruję na każdej stronie HTML, jest pierwsza funkcja Ajax w tej odpowiedzi, z tylko GET ...
Ale możesz zrobić wiele rzeczy za pomocą XMLHttpRequest 2:
Zrobiłem menedżera pobierania (używając zakresów po obu stronach z CV, filereader, system plików), różne konwertery resizerów obrazu za pomocą płótna, zapełniam internetowe bazy danych SQL obrazami base64 i wiele więcej ... Ale w takich przypadkach powinieneś stworzyć funkcję tylko do tego cel ... czasami potrzebujesz bloba, buforów tablicowych, możesz ustawić nagłówki, nadpisać typ MIME i jest o wiele więcej ...
Ale pytanie tutaj brzmi: jak zwrócić odpowiedź Ajax ... (dodałem łatwy sposób.)
Jeśli korzystasz z obietnic, ta odpowiedź jest dla Ciebie.
Oznacza to AngularJS, jQuery (z odroczeniem), natywną wymianę XHR (pobieranie), EmberJS, zapis BackboneJS lub dowolną bibliotekę węzłów, która zwraca obietnice.
Twój kod powinien być podobny do tego:
function foo() {
var data;
// or $.get(...).then, or request(...).then, or query(...).then
fetch("/echo/json").then(function(response){
data = response.json();
});
return data;
}
var result = foo(); // result is always undefined no matter what.
Felix Kling wykonał dobrą robotę, pisząc odpowiedź dla osób używających jQuery z wywołaniami zwrotnymi dla AJAX. Mam odpowiedź na natywny XHR. Ta odpowiedź dotyczy ogólnego użycia obietnic na frontend lub backend.
Podstawowa kwestia
Model współbieżności JavaScript w przeglądarce i na serwerze z NodeJS / io.js jest asynchroniczny i reaktywny .
Za każdym razem, gdy wywołujesz metodę, która zwraca obietnicę, procedury then
obsługi są zawsze wykonywane asynchronicznie - to znaczy po kodzie poniżej nich, którego nie ma w .then
procedurze obsługi.
Oznacza to, że kiedy zwracasz data
program then
obsługi, który zdefiniowałeś, nie został jeszcze wykonany. To z kolei oznacza, że zwracana wartość nie została ustawiona na poprawną wartość w czasie.
Oto prosta analogia do problemu:
function getFive(){
var data;
setTimeout(function(){ // set a timer for one second in the future
data = 5; // after a second, do this
}, 1000);
return data;
}
document.body.innerHTML = getFive(); // `undefined` here and not 5
Wartość data
jest, undefined
ponieważ data = 5
część nie została jeszcze wykonana. Prawdopodobnie zostanie wykonany w ciągu sekundy, ale do tego czasu nie będzie miał znaczenia dla zwracanej wartości.
Ponieważ operacja jeszcze się nie wydarzyła (AJAX, wywołanie serwera, IO, licznik czasu), zwracasz wartość, zanim żądanie będzie miało szansę powiedzieć kodowi, jaka to wartość.
Jednym z możliwych rozwiązań tego problemu jest ponowne aktywowanie kodu , informując program, co ma zrobić po zakończeniu obliczeń. Obietnice aktywnie to umożliwiają, będąc z natury czasowym (wrażliwym na czas).
Szybkie podsumowanie obietnic
Obietnica to wartość w czasie . Obietnice mają stan, zaczynają się jako oczekujące bez wartości i mogą zostać ustalone na:
- spełnione, co oznacza, że obliczenia zakończyły się pomyślnie.
- odrzucony, co oznacza, że obliczenia nie powiodły się.
Obietnica może zmienić stany tylko raz, po czym zawsze pozostanie w tym samym stanie na zawsze. Możesz dołączyć programy then
obsługi do obietnic w celu wyodrębnienia ich wartości i obsługi błędów. then
programy obsługi pozwalają na tworzenie łańcuchów połączeń. Obietnice są tworzone za pomocą interfejsów API, które je zwracają . Na przykład bardziej nowoczesny zamiennik AJAX fetch
lub $.get
obietnice zwrotu jQuery .
Kiedy zadzwonić .then
na obietnicy i powrotnej coś z tym - mamy obietnicę dla przetworzonego wartości . Jeśli zwrócimy kolejną obietnicę, dostaniemy niesamowite rzeczy, ale zatrzymajmy nasze konie.
Z obietnicami
Zobaczmy, jak możemy rozwiązać powyższy problem za pomocą obietnic. Najpierw pokażmy, jak rozumiemy stany obietnicy z góry, używając konstruktora Promise do tworzenia funkcji opóźnienia:
function delay(ms){ // takes amount of milliseconds
// returns a new promise
return new Promise(function(resolve, reject){
setTimeout(function(){ // when the time is up
resolve(); // change the promise to the fulfilled state
}, ms);
});
}
Teraz, po przekonwertowaniu setTimeout na używanie obietnic, możemy użyć, then
aby to się liczyło:
function delay(ms){ // takes amount of milliseconds
// returns a new promise
return new Promise(function(resolve, reject){
setTimeout(function(){ // when the time is up
resolve(); // change the promise to the fulfilled state
}, ms);
});
}
function getFive(){
// we're RETURNING the promise, remember, a promise is a wrapper over our value
return delay(100).then(function(){ // when the promise is ready
return 5; // return the value 5, promises are all about return values
})
}
// we _have_ to wrap it like this in the call site, we can't access the plain value
getFive().then(function(five){
document.body.innerHTML = five;
});
Zasadniczo, zamiast wrócić do wartości , której nie możemy zrobić, ponieważ model współbieżności - wracamy do otoki dla wartości, które możemy rozpakować z then
. To jak pudełko, którym możesz otworzyć then
.
Stosując to
To samo dotyczy Twojego oryginalnego wywołania API, możesz:
function foo() {
// RETURN the promise
return fetch("/echo/json").then(function(response){
return response.json(); // process it inside the `then`
});
}
foo().then(function(response){
// access the value inside the `then`
})
Więc to działa równie dobrze. Dowiedzieliśmy się, że nie możemy zwracać wartości z już asynchronicznych wywołań, ale możemy użyć obietnic i połączyć je w łańcuch w celu wykonania przetwarzania. Teraz wiemy, jak zwrócić odpowiedź z wywołania asynchronicznego.
ES2015 (ES6)
ES6 wprowadza generatory, które są funkcjami, które mogą powrócić w środku, a następnie wznowić punkt, w którym się znajdowali. Jest to zwykle przydatne w przypadku sekwencji, na przykład:
function* foo(){ // notice the star, this is ES6 so new browsers/node/io only
yield 1;
yield 2;
while(true) yield 3;
}
Jest funkcją, która zwraca iterator po sekwencji, 1,2,3,3,3,3,....
którą można iterować. Chociaż jest to interesujące samo w sobie i stwarza wiele możliwości, istnieje jeden szczególnie interesujący przypadek.
Jeśli sekwencja, którą tworzymy, jest sekwencją działań, a nie liczbami - możemy wstrzymać funkcję za każdym razem, gdy zostanie wykonana akcja i poczekać na nią, zanim ją wznowimy. A więc zamiast ciągu liczb potrzebujemy sekwencji przyszłych wartości - czyli: obietnic.
Ta nieco trudna, ale bardzo skuteczna sztuczka pozwala nam pisać kod asynchroniczny w sposób synchroniczny. Jest kilka „biegaczy”, którzy robią to za Ciebie. Napisanie jednego to kilka krótkich linii kodu, ale wykracza poza zakres tej odpowiedzi. Będę Promise.coroutine
tutaj używać Bluebird , ale są też inne opakowania, takie jak co
lub Q.async
.
var foo = coroutine(function*(){
var data = yield fetch("/echo/json"); // notice the yield
// code here only executes _after_ the request is done
return data.json(); // data is defined
});
Ta metoda sama zwraca obietnicę, którą możemy skonsumować z innych programów. Na przykład:
var main = coroutine(function*(){
var bar = yield foo(); // wait our earlier coroutine, it returns a promise
// server call done here, code below executes when done
var baz = yield fetch("/api/users/"+bar.userid); // depends on foo's result
console.log(baz); // runs after both requests done
});
main();
ES2016 (ES7)
W ES7 jest to bardziej ustandaryzowane, obecnie jest kilka propozycji, ale we wszystkich możesz await
obiecać. To jest po prostu „cukier” (ładniejsza składnia) dla powyższej propozycji ES6 poprzez dodanie słów kluczowych async
i await
. Robiąc powyższy przykład:
async function foo(){
var data = await fetch("/echo/json"); // notice the await
// code here only executes _after_ the request is done
return data.json(); // data is defined
}
Nadal zwraca obietnicę, tak samo :)
Niepoprawnie używasz Ajax. Chodzi o to, aby nie zwracał niczego, ale zamiast tego przekazał dane do czegoś, co nazywa się funkcją zwrotną, która obsługuje dane.
To jest:
function handleData( responseData ) {
// Do what you want with the data
console.log(responseData);
}
$.ajax({
url: "hi.php",
...
success: function ( data, status, XHR ) {
handleData(data);
}
});
Zwrócenie czegokolwiek w module obsługi przesyłania nic nie da. Zamiast tego musisz albo przekazać dane, albo zrobić z nimi, co chcesz, bezpośrednio w funkcji sukcesu.
Najprostszym rozwiązaniem jest utworzenie funkcji JavaScript i wywołanie jej dla success
wywołania zwrotnego Ajax .
function callServerAsync(){
$.ajax({ url: '...', success: function(response) { successCallback(response); } }); } function successCallback(responseObj){ // Do something like read the response and show data alert(JSON.stringify(responseObj)); // Only applicable to JSON response } function foo(callback) { $.ajax({
url: '...',
success: function(response) {
return callback(null, response);
}
});
}
var result = foo(function(err, result){
if (!err)
console.log(result);
});
Odpowiem okropnie wyglądającym, ręcznie rysowanym komiksem. Drugi obraz jest powodem, dla którego result
znajduje się undefined
w twoim przykładzie kodu.
Kątowy1
Osoby, które używają AngularJS , mogą sobie z tym poradzić używając Promises
.
Tutaj jest napisane:
Obietnice mogą być używane do niszczenia funkcji asynchronicznych i pozwalają łączyć wiele funkcji razem.
Można znaleźć ładne wyjaśnienie tutaj również.
Przykład znaleziony w dokumentach wymienionych poniżej.
promiseB = promiseA.then(
function onSuccess(result) {
return result + 1;
}
,function onError(err) {
//Handle error
}
);
// promiseB will be resolved immediately after promiseA is resolved
// and its value will be the result of promiseA incremented by 1.
Angular2 i później
W Angular2
ze spojrzeniem na poniższym przykładzie, ale jego zalecany do stosowania Observables
z Angular2
.
search(term: string) {
return this.http
.get(`https://api.spotify.com/v1/search?q=${term}&type=artist`)
.map((response) => response.json())
.toPromise();
}
Możesz to skonsumować w ten sposób,
search() {
this.searchService.search(this.searchField.value)
.then((result) => {
this.result = result.artists.items;
})
.catch((error) => console.error(error));
}
Zobacz oryginalny post tutaj. Ale Typescript nie obsługuje natywnych obietnic es6 , jeśli chcesz go używać, możesz potrzebować do tego wtyczki.
Dodatkowo tutaj jest specyfikacja obietnic zdefiniowana tutaj.
Większość odpowiedzi zawiera przydatne sugestie dotyczące pojedynczej operacji asynchronicznej, ale czasami pojawia się, gdy trzeba wykonać operację asynchroniczną dla każdego wpisu w tablicy lub innej strukturze podobnej do listy. Pokusa jest zrobienie tego:
// WRONG
var results = [];
theArray.forEach(function(entry) {
doSomethingAsync(entry, function(result) {
results.push(result);
});
});
console.log(results); // E.g., using them, returning them, etc.
Przykład:
// WRONG
var theArray = [1, 2, 3];
var results = [];
theArray.forEach(function(entry) {
doSomethingAsync(entry, function(result) {
results.push(result);
});
});
console.log("Results:", results); // E.g., using them, returning them, etc.
function doSomethingAsync(value, callback) {
console.log("Starting async operation for " + value);
setTimeout(function() {
console.log("Completing async operation for " + value);
callback(value * 2);
}, Math.floor(Math.random() * 200));
}
.as-console-wrapper {
max-height: 100% !important;
}
Przyczyną, która nie działa, jest to, że wywołania zwrotne od doSomethingAsync
nie zostały jeszcze uruchomione do czasu, gdy próbujesz wykorzystać wyniki.
Tak więc, jeśli masz tablicę (lub jakąś listę) i chcesz wykonać operacje asynchroniczne dla każdego wpisu, masz dwie opcje: Wykonaj operacje równolegle (nakładające się) lub szeregowo (jedna po drugiej w kolejności).
Równolegle
Możesz rozpocząć wszystkie z nich i śledzić, ile spodziewasz się wywołań zwrotnych, a następnie korzystać z wyników, gdy masz ich tyle:
var results = [];
var expecting = theArray.length;
theArray.forEach(function(entry, index) {
doSomethingAsync(entry, function(result) {
results[index] = result;
if (--expecting === 0) {
// Done!
console.log("Results:", results); // E.g., using the results
}
});
});
Przykład:
var theArray = [1, 2, 3];
var results = [];
var expecting = theArray.length;
theArray.forEach(function(entry, index) {
doSomethingAsync(entry, function(result) {
results[index] = result;
if (--expecting === 0) {
// Done!
console.log("Results:", results); // E.g., using the results
}
});
});
function doSomethingAsync(value, callback) {
console.log("Starting async operation for " + value);
setTimeout(function() {
console.log("Completing async operation for " + value);
callback(value * 2);
}, Math.floor(Math.random() * 200));
}
.as-console-wrapper {
max-height: 100% !important;
}
(Moglibyśmy pozbyć się expecting
i po prostu używać results.length === theArray.length
, ale to pozostawia nas otwartymi na możliwość theArray
zmiany, gdy połączenia są zaległe ...)
Zwróć uwagę, jak używamy index
from, forEach
aby zapisać wynik w results
tej samej pozycji, do której się odnosi, nawet jeśli wyniki przychodzą poza kolejnością (ponieważ wywołania asynchroniczne niekoniecznie kończą się w kolejności, w której zostały uruchomione).
Ale co, jeśli musisz zwrócić te wyniki z funkcji? Jak wskazywały inne odpowiedzi, nie możesz; musisz mieć funkcję akceptowania i wywoływania oddzwonienia (lub zwrotu Obietnicy ). Oto wersja wywołania zwrotnego:
function doSomethingWith(theArray, callback) {
var results = [];
var expecting = theArray.length;
theArray.forEach(function(entry, index) {
doSomethingAsync(entry, function(result) {
results[index] = result;
if (--expecting === 0) {
// Done!
callback(results);
}
});
});
}
doSomethingWith(theArray, function(results) {
console.log("Results:", results);
});
Przykład:
function doSomethingWith(theArray, callback) {
var results = [];
var expecting = theArray.length;
theArray.forEach(function(entry, index) {
doSomethingAsync(entry, function(result) {
results[index] = result;
if (--expecting === 0) {
// Done!
callback(results);
}
});
});
}
doSomethingWith([1, 2, 3], function(results) {
console.log("Results:", results);
});
function doSomethingAsync(value, callback) {
console.log("Starting async operation for " + value);
setTimeout(function() {
console.log("Completing async operation for " + value);
callback(value * 2);
}, Math.floor(Math.random() * 200));
}
.as-console-wrapper {
max-height: 100% !important;
}
Lub tutaj jest wersja zwracająca Promise
zamiast tego:
function doSomethingWith(theArray) {
return new Promise(function(resolve) {
var results = [];
var expecting = theArray.length;
theArray.forEach(function(entry, index) {
doSomethingAsync(entry, function(result) {
results[index] = result;
if (--expecting === 0) {
// Done!
resolve(results);
}
});
});
});
}
doSomethingWith(theArray).then(function(results) {
console.log("Results:", results);
});
Oczywiście, gdybyśmy doSomethingAsync
przekazali nam błędy, reject
odrzucilibyśmy obietnicę, gdy otrzymaliśmy błąd).
Przykład:
function doSomethingWith(theArray) {
return new Promise(function(resolve) {
var results = [];
var expecting = theArray.length;
theArray.forEach(function(entry, index) {
doSomethingAsync(entry, function(result) {
results[index] = result;
if (--expecting === 0) {
// Done!
resolve(results);
}
});
});
});
}
doSomethingWith([1, 2, 3]).then(function(results) {
console.log("Results:", results);
});
function doSomethingAsync(value, callback) {
console.log("Starting async operation for " + value);
setTimeout(function() {
console.log("Completing async operation for " + value);
callback(value * 2);
}, Math.floor(Math.random() * 200));
}
.as-console-wrapper {
max-height: 100% !important;
}
(Lub alternatywnie możesz zrobić opakowanie, doSomethingAsync
które zwraca obietnicę, a następnie wykonaj poniższe czynności ...)
Jeśli doSomethingAsync
daje Ci Obietnicę , możesz użyć Promise.all:
function doSomethingWith(theArray) {
return Promise.all(theArray.map(function(entry) {
return doSomethingAsync(entry);
}));
}
doSomethingWith(theArray).then(function(results) {
console.log("Results:", results);
});
Jeśli wiesz, że doSomethingAsync
zignoruje drugi i trzeci argument, możesz po prostu przekazać go bezpośrednio map
( map
wywołuje wywołanie zwrotne z trzema argumentami, ale większość ludzi używa tylko pierwszego):
function doSomethingWith(theArray) {
return Promise.all(theArray.map(doSomethingAsync));
}
doSomethingWith(theArray).then(function(results) {
console.log("Results:", results);
});
Przykład:
function doSomethingWith(theArray) {
return Promise.all(theArray.map(doSomethingAsync));
}
doSomethingWith([1, 2, 3]).then(function(results) {
console.log("Results:", results);
});
function doSomethingAsync(value) {
console.log("Starting async operation for " + value);
return new Promise(function(resolve) {
setTimeout(function() {
console.log("Completing async operation for " + value);
resolve(value * 2);
}, Math.floor(Math.random() * 200));
});
}
.as-console-wrapper {
max-height: 100% !important;
}
Zwróć uwagę, że Promise.all
wypełnia swoją obietnicę z szeregiem wyników wszystkich obietnic, które jej dałeś, gdy wszystkie zostaną rozwiązane, lub odrzuca swoją obietnicę, gdy pierwsza z obietnic, które dałeś, odrzuca.
Seria
Przypuśćmy, że nie chcesz, aby operacje były równoległe? Jeśli chcesz uruchamiać je jedna po drugiej, musisz poczekać na zakończenie każdej operacji, zanim zaczniesz następną. Oto przykład funkcji, która to robi i wywołuje wywołanie zwrotne z wynikiem:
function doSomethingWith(theArray, callback) {
var results = [];
doOne(0);
function doOne(index) {
if (index < theArray.length) {
doSomethingAsync(theArray[index], function(result) {
results.push(result);
doOne(index + 1);
});
} else {
// Done!
callback(results);
}
}
}
doSomethingWith(theArray, function(results) {
console.log("Results:", results);
});
(Ponieważ pracujemy w serii, możemy po prostu użyć, results.push(result)
ponieważ wiemy, że nie uzyskamy wyników w kolejności. W powyższym moglibyśmy użyć results[index] = result;
, ale w niektórych z poniższych przykładów nie mamy indeksu używać.)
Przykład:
function doSomethingWith(theArray, callback) {
var results = [];
doOne(0);
function doOne(index) {
if (index < theArray.length) {
doSomethingAsync(theArray[index], function(result) {
results.push(result);
doOne(index + 1);
});
} else {
// Done!
callback(results);
}
}
}
doSomethingWith([1, 2, 3], function(results) {
console.log("Results:", results);
});
function doSomethingAsync(value, callback) {
console.log("Starting async operation for " + value);
setTimeout(function() {
console.log("Completing async operation for " + value);
callback(value * 2);
}, Math.floor(Math.random() * 200));
}
.as-console-wrapper {
max-height: 100% !important;
}
(Lub, znowu, utwórz opakowanie, doSomethingAsync
które daje ci obietnicę i wykonaj poniższe czynności ...)
Jeśli doSomethingAsync
daje Ci Obietnicę, jeśli możesz użyć składni ES2017 + (być może z transpilerem, takim jak Babel ), możesz użyć asyncfunkcji z for-ofi await:
async function doSomethingWith(theArray) {
const results = [];
for (const entry of theArray) {
results.push(await doSomethingAsync(entry));
}
return results;
}
doSomethingWith(theArray).then(results => {
console.log("Results:", results);
});
Przykład:
async function doSomethingWith(theArray) {
const results = [];
for (const entry of theArray) {
results.push(await doSomethingAsync(entry));
}
return results;
}
doSomethingWith([1, 2, 3]).then(function(results) {
console.log("Results:", results);
});
function doSomethingAsync(value) {
console.log("Starting async operation for " + value);
return new Promise(function(resolve) {
setTimeout(function() {
console.log("Completing async operation for " + value);
resolve(value * 2);
}, Math.floor(Math.random() * 200));
});
}
.as-console-wrapper {
max-height: 100% !important;
}
Jeśli nie możesz użyć składni ES2017 + (jeszcze), możesz użyć odmiany wzorca „Obietnica redukuj” (jest to bardziej złożone niż zwykła redukcja Obietnicy, ponieważ nie przekazujemy wyniku z jednego do drugiego, ale zamiast tego gromadzenie ich wyników w tablicy):
function doSomethingWith(theArray) {
return theArray.reduce(function(p, entry) {
return p.then(function(results) {
return doSomethingAsync(entry).then(function(result) {
results.push(result);
return results;
});
});
}, Promise.resolve([]));
}
doSomethingWith(theArray).then(function(results) {
console.log("Results:", results);
});
Przykład:
function doSomethingWith(theArray) {
return theArray.reduce(function(p, entry) {
return p.then(function(results) {
return doSomethingAsync(entry).then(function(result) {
results.push(result);
return results;
});
});
}, Promise.resolve([]));
}
doSomethingWith([1, 2, 3]).then(function(results) {
console.log("Results:", results);
});
function doSomethingAsync(value) {
console.log("Starting async operation for " + value);
return new Promise(function(resolve) {
setTimeout(function() {
console.log("Completing async operation for " + value);
resolve(value * 2);
}, Math.floor(Math.random() * 200));
});
}
.as-console-wrapper {
max-height: 100% !important;
}
... co jest mniej kłopotliwe dzięki funkcjom strzałkowym ES2015 + :
function doSomethingWith(theArray) {
return theArray.reduce((p, entry) => p.then(results => doSomethingAsync(entry).then(result => {
results.push(result);
return results;
})), Promise.resolve([]));
}
doSomethingWith(theArray).then(results => {
console.log("Results:", results);
});
Przykład:
function doSomethingWith(theArray) {
return theArray.reduce((p, entry) => p.then(results => doSomethingAsync(entry).then(result => {
results.push(result);
return results;
})), Promise.resolve([]));
}
doSomethingWith([1, 2, 3]).then(function(results) {
console.log("Results:", results);
});
function doSomethingAsync(value) {
console.log("Starting async operation for " + value);
return new Promise(function(resolve) {
setTimeout(function() {
console.log("Completing async operation for " + value);
resolve(value * 2);
}, Math.floor(Math.random() * 200));
});
}
.as-console-wrapper {
max-height: 100% !important;
}
Spójrz na ten przykład:
var app = angular.module('plunker', []);
app.controller('MainCtrl', function($scope,$http) {
var getJoke = function(){
return $http.get('http://api.icndb.com/jokes/random').then(function(res){
return res.data.value;
});
}
getJoke().then(function(res) {
console.log(res.joke);
});
});
Jak widzisz, getJoke
jest to zwrot rozwiązanej obietnicy (jest ona rozpatrywana podczas zwrotu res.data.value
). Więc czekasz, aż żądanie $ http.get zostanie zakończone, a następnie zostanie wykonany console.log (res.joke) (jako normalny przepływ asynchroniczny).
To jest plnkr:
http://embed.plnkr.co/XlNR7HpCaIhJxskMJfSg/
Droga ES6 (async - czekaj)
(function(){
async function getJoke(){
let response = await fetch('http://api.icndb.com/jokes/random');
let data = await response.json();
return data.value;
}
getJoke().then((joke) => {
console.log(joke);
});
})();
Jest to jedno z miejsc, w których dwa sposoby wiązania danych lub koncepcji przechowywania, które są używane w wielu nowych frameworkach JavaScript, będą dla Ciebie świetne ...
Więc jeśli używasz Angular, React lub jakichkolwiek innych frameworków, które wykonują dwa sposoby wiązania danych lub koncepcję przechowywania, ten problem jest po prostu rozwiązany, więc w prostych słowach twój wynik jest undefined
na pierwszym etapie, więc masz result = undefined
przed otrzymaniem dane, to jak tylko otrzymasz wynik, zostanie zaktualizowany i zostanie przypisany do nowej wartości, która odpowiedź na twoje wywołanie Ajax ...
Ale jak możesz to zrobić w czystym javascript lub jQuery, na przykład, jak zadałeś w tym pytaniu?
Możesz użyć wywołania zwrotnego , obietnicy i ostatnio obserwowalnych, aby obsłużyć to za Ciebie, na przykład w obietnicach mamy jakąś funkcję podobną success()
lub then()
która zostanie wykonana, gdy twoje dane będą dla Ciebie gotowe, tak samo z funkcją oddzwaniania lub subskrypcji na obserable .
Na przykład w Twoim przypadku, w którym używasz jQuery , możesz zrobić coś takiego:
$(document).ready(function(){ function foo() { $.ajax({url: "api/data", success: function(data){
fooDone(data); //after we have data, we pass it to fooDone
}});
};
function fooDone(data) {
console.log(data); //fooDone has the data and console.log it
};
foo(); //call happens here
});
Aby uzyskać więcej informacji, zapoznaj się z obietnicami i obserwacjami, które są nowszymi sposobami wykonywania tych czynności asynchronicznych.
Jest to bardzo częsty problem, z którym borykamy się podczas zmagania się z „tajemnicami” JavaScript. Spróbuję dziś zdemistyfikować tę tajemnicę.
Zacznijmy od prostej funkcji JavaScript:
function foo(){
// do something
return 'wohoo';
}
let bar = foo(); // bar is 'wohoo' here
To proste wywołanie funkcji synchronicznej (gdzie każdy wiersz kodu jest „zakończony swoim zadaniem” przed następnym w kolejności), a wynik jest taki sam, jak oczekiwano.
Teraz dodajmy trochę skrętu, wprowadzając niewielkie opóźnienie w naszej funkcji, aby wszystkie wiersze kodu nie były „kończone” w sekwencji. W ten sposób będzie emulować asynchroniczne zachowanie funkcji:
function foo(){
setTimeout( ()=>{
return 'wohoo';
}, 1000 )
}
let bar = foo() // bar is undefined here
Więc proszę bardzo, to opóźnienie po prostu zepsuło funkcjonalność, której się spodziewaliśmy! Ale co dokładnie się stało? Cóż, jest to całkiem logiczne, jeśli spojrzysz na kod. funkcja foo()
po wykonaniu nic nie zwraca (w ten sposób zwracana jest wartość undefined
), ale uruchamia licznik czasu, który wykonuje funkcję po 1 s, aby zwrócić „wohoo”. Ale jak widzisz, wartość przypisana do bar jest natychmiast zwróconą przez foo (), co jest niczym, tj undefined
. Po prostu .
Jak więc rozwiązujemy ten problem?
Zapytajmy naszą funkcję o OBIETNICĘ . Obietnica tak naprawdę dotyczy tego, co to znaczy: oznacza, że funkcja gwarantuje dostarczenie wszelkich wyników, jakie otrzyma w przyszłości. zobaczmy więc, jak to działa w przypadku naszego małego problemu powyżej:
function foo(){
return new Promise( (resolve, reject) => { // I want foo() to PROMISE me something
setTimeout ( function(){
// promise is RESOLVED , when execution reaches this line of code
resolve('wohoo')// After 1 second, RESOLVE the promise with value 'wohoo'
}, 1000 )
})
}
let bar ;
foo().then( res => {
bar = res;
console.log(bar) // will print 'wohoo'
});
Zatem podsumowanie jest takie - aby poradzić sobie z funkcjami asynchronicznymi, takimi jak wywołania oparte na Ajax itp., Możesz użyć obietnicy resolve
wartości (którą zamierzasz zwrócić). Tak więc, w skrócie , w funkcjach asynchronicznych rozwiązujesz wartość zamiast zwracać .
UPDATE (Promises with async / await)
Oprócz wykorzystywania then/catch
obietnic do pracy, istnieje jeszcze jedno podejście. Chodzi o to, aby rozpoznać funkcję asynchroniczną, a następnie poczekać na rozwiązanie obietnic , przed przejściem do następnego wiersza kodu. To wciąż tylko promises
pod maską, ale z innym podejściem syntaktycznym. Aby wszystko było jaśniejsze, poniżej możesz znaleźć porównanie:
następnie / catch wersja:
function saveUsers(){
getUsers()
.then(users => {
saveSomewhere(users);
})
.catch(err => {
console.error(err);
})
}
wersja async / await:
async function saveUsers(){
try{
let users = await getUsers()
saveSomewhere(users);
}
catch(err){
console.error(err);
}
}
Innym podejściem do zwracania wartości z funkcji asynchronicznej jest przekazanie obiektu, który będzie przechowywać wynik funkcji asynchronicznej.
Oto przykład tego samego:
var async = require("async");
// This wires up result back to the caller
var result = {};
var asyncTasks = [];
asyncTasks.push(function(_callback){
// some asynchronous operation
$.ajax({
url: '...',
success: function(response) {
result.response = response;
_callback();
}
});
});
async.parallel(asyncTasks, function(){
// result is available after performing asynchronous operation
console.log(result)
console.log('Done');
});
Używam result
obiektu do przechowywania wartości podczas operacji asynchronicznej. Dzięki temu wynik będzie dostępny nawet po wykonaniu zadania asynchronicznego.
Często używam tego podejścia. Chciałbym wiedzieć, jak dobrze działa to podejście, gdy wymagane jest okablowanie wyniku z powrotem przez kolejne moduły.
Chociaż obietnice i wezwania zwrotne działają dobrze w wielu sytuacjach, uciążliwe jest wyrażenie czegoś takiego:
if (!name) {
name = async1();
}
async2(name);
Skończysz przez to async1
; sprawdź, czy name
jest niezdefiniowana, czy nie, i odpowiednio wywołaj callback.
async1(name, callback) {
if (name)
callback(name)
else {
doSomething(callback)
}
}
async1(name, async2)
Chociaż w małych przykładach jest to w porządku , staje się irytujące, gdy masz wiele podobnych przypadków i związanych z obsługą błędów.
Fibers
pomaga w rozwiązaniu problemu.
var Fiber = require('fibers')
function async1(container) {
var current = Fiber.current
var result
doSomething(function(name) {
result = name
fiber.run()
})
Fiber.yield()
return result
}
Fiber(function() {
var name
if (!name) {
name = async1()
}
async2(name)
// Make any number of async calls from here
}
Możesz sprawdzić projekt tutaj .
Poniższy przykład, który napisałem, pokazuje, jak to zrobić
- Obsługuj asynchroniczne wywołania HTTP;
- Czekaj na odpowiedź z każdego wywołania API;
- Użyj wzoru Obietnica ;
- Użyj wzorca Promise.all , aby dołączyć do wielu wywołań HTTP;
Ten przykład roboczy jest niezależny. Zdefiniuje prosty obiekt żądania, który używa XMLHttpRequest
obiektu okna do wykonywania połączeń. Zdefiniuje prostą funkcję czekającą na wypełnienie kilku obietnic.
Kontekst. Przykładem jest wysłanie zapytania do punktu końcowego Spotify Web API w celu wyszukania playlist
obiektów dla danego zestawu ciągów zapytań:
[
"search?type=playlist&q=%22doom%20metal%22",
"search?type=playlist&q=Adele"
]
Dla każdego elementu nowa Promise uruchomi blok - ExecutionBlock
przeanalizuje wynik, zaplanuje nowy zestaw obietnic w oparciu o tablicę wyników, czyli listę user
obiektów Spotify, i ExecutionProfileBlock
asynchronicznie wykona nowe wywołanie HTTP .
Następnie możesz zobaczyć zagnieżdżoną strukturę Promise, która umożliwia tworzenie wielu i całkowicie asynchronicznych zagnieżdżonych wywołań HTTP i łączenie wyników z każdego podzbioru wywołań Promise.all
.
UWAGA Najnowsze search
interfejsy API Spotify będą wymagać podania tokena dostępu w nagłówkach żądań:
-H "Authorization: Bearer {your access token}"
Aby więc uruchomić następujący przykład, musisz umieścić swój token dostępu w nagłówkach żądań:
var spotifyAccessToken = "YourSpotifyAccessToken";
var console = {
log: function(s) {
document.getElementById("console").innerHTML += s + "<br/>"
}
}
// Simple XMLHttpRequest
// based on https://davidwalsh.name/xmlhttprequest
SimpleRequest = {
call: function(what, response) {
var request;
if (window.XMLHttpRequest) { // Mozilla, Safari, ...
request = new XMLHttpRequest();
} else if (window.ActiveXObject) { // Internet Explorer
try {
request = new ActiveXObject('Msxml2.XMLHTTP');
}
catch (e) {
try {
request = new ActiveXObject('Microsoft.XMLHTTP');
} catch (e) {}
}
}
// State changes
request.onreadystatechange = function() {
if (request.readyState === 4) { // Done
if (request.status === 200) { // Complete
response(request.responseText)
}
else
response();
}
}
request.open('GET', what, true);
request.setRequestHeader("Authorization", "Bearer " + spotifyAccessToken);
request.send(null);
}
}
//PromiseAll
var promiseAll = function(items, block, done, fail) {
var self = this;
var promises = [],
index = 0;
items.forEach(function(item) {
promises.push(function(item, i) {
return new Promise(function(resolve, reject) {
if (block) {
block.apply(this, [item, index, resolve, reject]);
}
});
}(item, ++index))
});
Promise.all(promises).then(function AcceptHandler(results) {
if (done) done(results);
}, function ErrorHandler(error) {
if (fail) fail(error);
});
}; //promiseAll
// LP: deferred execution block
var ExecutionBlock = function(item, index, resolve, reject) {
var url = "https://api.spotify.com/v1/"
url += item;
console.log( url )
SimpleRequest.call(url, function(result) {
if (result) {
var profileUrls = JSON.parse(result).playlists.items.map(function(item, index) {
return item.owner.href;
})
resolve(profileUrls);
}
else {
reject(new Error("call error"));
}
})
}
arr = [
"search?type=playlist&q=%22doom%20metal%22",
"search?type=playlist&q=Adele"
]
promiseAll(arr, function(item, index, resolve, reject) {
console.log("Making request [" + index + "]")
ExecutionBlock(item, index, resolve, reject);
}, function(results) { // Aggregated results
console.log("All profiles received " + results.length);
//console.log(JSON.stringify(results[0], null, 2));
///// promiseall again
var ExecutionProfileBlock = function(item, index, resolve, reject) {
SimpleRequest.call(item, function(result) {
if (result) {
var obj = JSON.parse(result);
resolve({
name: obj.display_name,
followers: obj.followers.total,
url: obj.href
});
} //result
})
} //ExecutionProfileBlock
promiseAll(results[0], function(item, index, resolve, reject) {
//console.log("Making request [" + index + "] " + item)
ExecutionProfileBlock(item, index, resolve, reject);
}, function(results) { // aggregated results
console.log("All response received " + results.length);
console.log(JSON.stringify(results, null, 2));
}
, function(error) { // Error
console.log(error);
})
/////
},
function(error) { // Error
console.log(error);
});
<div id="console" />
Obszernie omówiłem tutaj to rozwiązanie .
Krótka odpowiedź brzmi: musisz zaimplementować takie wywołanie zwrotne:
function callback(response) {
// Here you can do what ever you want with the response object.
console.log(response);
}
$.ajax({
url: "...",
success: callback
});
Odpowiedź z 2017 roku: możesz teraz robić dokładnie to, co chcesz w każdej aktualnej przeglądarce i węźle
To jest dość proste:
- Zwróć obietnicę
- Użyj `` await '' , które powie JavaScriptowi, aby czekał na obietnicę przekształcenia w wartość (jak odpowiedź HTTP)
- Dodaj słowo kluczowe „async” do funkcji nadrzędnej
Oto działająca wersja Twojego kodu:
(async function(){
var response = await superagent.get('...')
console.log(response)
})()
await jest obsługiwany we wszystkich obecnych przeglądarkach i węźle 8
Js jest jednowątkowy.
Przeglądarkę można podzielić na trzy części:
1) Pętla zdarzeń
2) Web API
3) Kolejka wydarzeń
Pętla zdarzeń działa na zawsze, tzn. Jest to rodzaj nieskończonej pętli. Kolejka zdarzeń to miejsce, w którym cała Twoja funkcja jest wypychana na jakieś zdarzenie (na przykład: kliknięcie), jest to jedno po drugim wykonywane z kolejki i umieszczane w pętli zdarzeń, która wykonuje tę funkcję i przygotowuje ją samodzielnie dla następnej po wykonaniu pierwszej funkcji, co oznacza, że wykonanie jednej funkcji nie rozpocznie się, dopóki funkcja znajdująca się przed nią w kolejce nie zostanie wykonana w pętli zdarzeń.
Teraz pomyślmy, że umieściliśmy dwie funkcje w kolejce, jedna służy do pobierania danych z serwera, a druga wykorzystuje te dane. Najpierw umieściliśmy w kolejce funkcję serverRequest (), a następnie funkcję utiliseData (). Funkcja serverRequest przechodzi w pętlę zdarzeń i wywołuje serwer, ponieważ nigdy nie wiemy, ile czasu zajmie pobranie danych z serwera, więc oczekuje się, że ten proces zajmie trochę czasu, więc zajmiemy się naszą pętlą zdarzeń, w ten sposób zawieszając naszą stronę. API wchodzi w rolę, bierze tę funkcję z pętli zdarzeń i zajmuje się serwerem zwalniającym pętlę zdarzeń, abyśmy mogli wykonać następną funkcję z kolejki Następną funkcją w kolejce jest utiliseData (), która przechodzi w pętlę, ale z powodu braku danych przechodzi marnowanie i wykonywanie następnej funkcji trwa do końca kolejki (nazywa się to wywoływaniem Async, czyli możemy robić coś innego, aż otrzymamy dane)
Załóżmy, że nasza funkcja serverRequest () miała instrukcję return w kodzie, kiedy otrzymamy dane z serwera, API sieci Web wepchnie je do kolejki na końcu kolejki. Ponieważ jest on wypychany na końcu w kolejce, nie możemy wykorzystać jego danych, ponieważ w naszej kolejce nie ma już żadnej funkcji do wykorzystania tych danych. Dlatego nie jest możliwe zwrócenie czegoś z wywołania asynchronicznego.
Tak więc rozwiązaniem tego jest callback lub obietnica .
Obraz z jednej z odpowiedzi tutaj, Poprawnie wyjaśnia użycie wywołania zwrotnego ... Dajemy naszą funkcję (funkcję wykorzystującą dane zwrócone z serwera) do funkcji wywołującej serwer.
function doAjax(callbackFunc, method, url) {
var xmlHttpReq = new XMLHttpRequest();
xmlHttpReq.open(method, url);
xmlHttpReq.onreadystatechange = function() {
if (xmlHttpReq.readyState == 4 && xmlHttpReq.status == 200) {
callbackFunc(xmlHttpReq.responseText);
}
}
xmlHttpReq.send(null);
}
W moim kodzie nazywa się to
function loadMyJson(categoryValue){
if(categoryValue==="veg")
doAjax(print,"GET","http://localhost:3004/vegetables");
else if(categoryValue==="fruits")
doAjax(print,"GET","http://localhost:3004/fruits");
else
console.log("Data not found");
}
Wywołanie zwrotne Javscript.info
Możesz użyć tej własnej biblioteki (napisanej przy użyciu Promise) do wykonania zdalnego połączenia.
function $http(apiConfig) {
return new Promise(function (resolve, reject) {
var client = new XMLHttpRequest();
client.open(apiConfig.method, apiConfig.url);
client.send();
client.onload = function () {
if (this.status >= 200 && this.status < 300) {
// Performs the function "resolve" when this.status is equal to 2xx.
// Your logic here.
resolve(this.response);
}
else {
// Performs the function "reject" when this.status is different than 2xx.
reject(this.statusText);
}
};
client.onerror = function () {
reject(this.statusText);
};
});
}
Prosty przykład użycia:
$http({
method: 'get',
url: 'google.com'
}).then(function(response) {
console.log(response);
}, function(error) {
console.log(error)
});
Innym rozwiązaniem jest wykonanie kodu za pomocą sekwencyjnego modułu wykonawczego nsynjs .
Jeśli podstawowa funkcja jest obiecana
nsynjs oceni wszystkie obietnice po kolei i umieści wynik obietnicy we data
właściwości:
function synchronousCode() {
var getURL = function(url) {
return window.fetch(url).data.text().data;
};
var url = 'https://ajax.googleapis.com/ajax/libs/jquery/2.0.0/jquery.min.js';
console.log('received bytes:',getURL(url).length);
};
nsynjs.run(synchronousCode,{},function(){
console.log('synchronousCode done');
});
<script src="https://rawgit.com/amaksr/nsynjs/master/nsynjs.js"></script>
Jeśli podstawowa funkcja nie jest obiecana
Krok 1. Umieść funkcję z wywołaniem zwrotnym w opakowaniu obsługującym nsynjs (jeśli ma obiecaną wersję, możesz pominąć ten krok):
var ajaxGet = function (ctx,url) {
var res = {};
var ex;
$.ajax(url)
.done(function (data) {
res.data = data;
})
.fail(function(e) {
ex = e;
})
.always(function() {
ctx.resume(ex);
});
return res;
};
ajaxGet.nsynjsHasCallback = true;
Krok 2. Umieść logikę synchroniczną w funkcji:
function process() {
console.log('got data:', ajaxGet(nsynjsCtx, "data/file1.json").data);
}
Krok 3. Uruchom funkcję w sposób synchroniczny przez nsynjs:
nsynjs.run(process,this,function () {
console.log("synchronous function finished");
});
Nsynjs oceni wszystkie operatory i wyrażenia krok po kroku, wstrzymując wykonywanie w przypadku, gdy wynik jakiejś wolnej funkcji nie jest gotowy.
Więcej przykładów tutaj: https://github.com/amaksr/nsynjs/tree/master/examples
ECMAScript 6 posiada „generatory”, które umożliwiają łatwe programowanie w stylu asynchronicznym.
function* myGenerator() {
const callback = yield;
let [response] = yield $.ajax("https://stackoverflow.com", {complete: callback});
console.log("response is:", response);
// examples of other things you can do
yield setTimeout(callback, 1000);
console.log("it delayed for 1000ms");
while (response.statusText === "error") {
[response] = yield* anotherGenerator();
}
}
Aby uruchomić powyższy kod, wykonaj następujące czynności:
const gen = myGenerator(); // Create generator
gen.next(); // Start it
gen.next((...args) => gen.next([...args])); // Set its callback function
Jeśli chcesz wybrać przeglądarki, które nie obsługują ES6, możesz uruchomić kod za pomocą Babel lub kompilatora zamknięcia, aby wygenerować ECMAScript 5.
Wywołania zwrotne ...args
są opakowane w tablicę i niszczone podczas ich czytania, dzięki czemu wzorzec poradzi sobie z wywołaniami zwrotnymi, które mają wiele argumentów. Na przykład z węzłem fs :
const [err, data] = yield fs.readFile(filePath, "utf-8", callback);
Oto kilka podejść do pracy z żądaniami asynchronicznymi:
- Browser Promise
- P: Biblioteka obietnic dla JavaScript
- A + Promises.js
- jQuery odroczony
- Interfejs API XMLHttpRequest
- Korzystanie z koncepcji oddzwaniania - jako implementacja w pierwszej odpowiedzi
Przykład: implementacja jQuery odroczona do pracy z wieloma żądaniami
var App = App || {};
App = {
getDataFromServer: function(){
var self = this,
deferred = $.Deferred(), requests = []; requests.push($.getJSON('request/ajax/url/1'));
requests.push($.getJSON('request/ajax/url/2')); $.when.apply(jQuery, requests).done(function(xhrResponse) {
return deferred.resolve(xhrResponse.result);
});
return deferred;
},
init: function(){
this.getDataFromServer().done(_.bind(function(resp1, resp2) {
// Do the operations which you wanted to do when you
// get a response from Ajax, for example, log response.
}, this));
}
};
App.init();
Znajdujemy się we wszechświecie, który wydaje się postępować w wymiarze, który nazywamy „czasem”. Naprawdę nie rozumiemy, czym jest czas, ale opracowaliśmy abstrakcje i słownictwo, które pozwala nam rozumować i mówić o nim: „przeszłość”, „teraźniejszość”, „przyszłość”, „przed”, „po”.
Budowane przez nas systemy komputerowe - coraz częściej - mają czas jako ważny wymiar. Pewne rzeczy mają wydarzyć się w przyszłości. Potem inne rzeczy muszą się wydarzyć, gdy te pierwsze rzeczy w końcu się wydarzą. To jest podstawowe pojęcie zwane „asynchronicznością”. W naszym coraz bardziej sieciowym świecie najczęstszym przypadkiem asynchroniczności jest oczekiwanie, aż jakiś zdalny system odpowie na jakieś żądanie.
Rozważmy przykład. Dzwonisz do mleczarza i zamawiasz mleko. Jeśli chodzi o to, chcesz dodać go do swojej kawy. Nie możesz teraz wlać mleka do kawy, ponieważ jeszcze go nie ma. Musisz poczekać, aż nadejdzie, zanim włożysz go do kawy. Innymi słowy, nie zadziała:
var milk = order_milk();
put_in_coffee(milk);
Ponieważ JS nie ma możliwości dowiedzenia się, że musi czekać na order_milk
zakończenie, zanim się wykona put_in_coffee
. Innymi słowy, nie wie, że order_milk
jest asynchroniczny - to coś, co skutkuje mlekiem dopiero w przyszłości. JS i inne języki deklaratywne wykonują jedną instrukcję po drugiej bez czekania.
Klasyczne podejście JS do tego problemu, wykorzystujące fakt, że JS obsługuje funkcje jako obiekty pierwszej klasy, które można przekazywać, polega na przekazaniu funkcji jako parametru do żądania asynchronicznego, które następnie wywoła po zakończeniu jego zadanie w przyszłości. To jest podejście „oddzwonienia”. To wygląda tak:
order_milk(put_in_coffee);
order_milk
uruchamia się, zamawia mleko, a potem, kiedy i tylko kiedy nadejdzie, przywołuje put_in_coffee
.
Problem z tym podejściem wywołania zwrotnego polega na tym, że zanieczyszcza ono normalną semantykę funkcji raportującej swój wynik return
; zamiast tego funkcje nie mogą zgłaszać swoich wyników przez wywołanie wywołania zwrotnego podanego jako parametr. Ponadto podejście to może szybko stać się nieporęczne w przypadku dłuższych sekwencji wydarzeń. Np. Powiedzmy, że chcę poczekać, aż mleko wleje się do kawy, a potem i dopiero wtedy wykonać trzeci krok, czyli wypicie kawy. W końcu muszę napisać coś takiego:
order_milk(function(milk) { put_in_coffee(milk, drink_coffee); }
gdzie przechodzę put_in_coffee
zarówno do mleka, aby w nim umieścić, jak i do akcji ( drink_coffee
) do wykonania po włożeniu mleka. Taki kod staje się trudny do napisania, odczytania i debugowania.
W takim przypadku moglibyśmy przepisać kod w pytaniu jako:
var answer;
$.ajax('/foo.json') . done(function(response) {
callback(response.data);
});
function callback(data) {
console.log(data);
}
Wprowadź obietnice
To była motywacja do pojęcia „obietnicy”, która jest szczególnym rodzajem wartości, która reprezentuje jakiś przyszły lub asynchroniczny wynik. Może reprezentować coś, co już się wydarzyło, co wydarzy się w przyszłości lub może nigdy się nie wydarzyć. Obietnice mają jedną metodę, nazwaną then
, do której przekazujesz akcję do wykonania, gdy wynik, który reprezentuje obietnica, zostanie zrealizowany.
W przypadku naszego mleka i kawy projektujemy order_milk
zwrot obietnicy za przybycie mleka, a następnie określamy put_in_coffee
jako then
czynność w następujący sposób:
order_milk() . then(put_in_coffee)
Jedną z zalet tego jest to, że możemy je łączyć ze sobą, aby tworzyć sekwencje przyszłych zdarzeń („łańcuch”):
order_milk() . then(put_in_coffee) . then(drink_coffee)
Zastosujmy obietnice do twojego konkretnego problemu. Naszą logikę żądania zawiniemy w funkcję, która zwraca obietnicę:
function get_data() {
return $.ajax('/foo.json');
}
Właściwie wszystko, co zrobiliśmy, to dodanie return
do wywołania $.ajax
. To działa, ponieważ jQuery $.ajax
już zwraca coś w rodzaju obietnicy. (W praktyce, bez wchodzenia w szczegóły, wolelibyśmy zawinąć to wywołanie, aby zwrócić prawdziwą obietnicę, lub użyć innej alternatywy $.ajax
.) Teraz, jeśli chcemy załadować plik i poczekać, aż się zakończy i następnie zrób coś, możemy po prostu powiedzieć
get_data() . then(do_something)
na przykład,
get_data() .
then(function(data) { console.log(data); });
Korzystając z obietnic, przekazujemy wiele funkcji do programu then
, więc często pomocne jest użycie bardziej kompaktowych funkcji strzałkowych w stylu ES6:
get_data() .
then(data => console.log(data));
Słowo async
kluczowe
Ale wciąż jest coś niejasno niezadowalającego w konieczności pisania kodu w jeden sposób, jeśli jest synchroniczny, a zupełnie inny, jeśli jest asynchroniczny. W przypadku synchronicznych piszemy
a();
b();
ale jeśli a
jest asynchroniczny, z obietnicami musimy pisać
a() . then(b);
Powyżej powiedzieliśmy: „JS nie ma możliwości dowiedzenia się, że musi czekać na zakończenie pierwszego wywołania, zanim wykona drugie”. Czy nie byłoby miło, gdyby tam był jakiś sposób, aby powiedzieć, że JS? Okazuje się, że istnieje - await
słowo kluczowe, używane wewnątrz specjalnego typu funkcji zwanej funkcją „async”. Ta funkcja jest częścią nadchodzącej wersji ES, ale jest już dostępna w transpilerach, takich jak Babel, mając odpowiednie ustawienia wstępne. To pozwala nam po prostu pisać
async function morning_routine() {
var milk = await order_milk();
var coffee = await put_in_coffee(milk);
await drink(coffee);
}
W twoim przypadku byłbyś w stanie napisać coś takiego
async function foo() {
data = await get_data();
console.log(data);
}
Krótka odpowiedź : Twoja foo()
metoda zwraca natychmiast, podczas gdy $ajax()
wywołanie jest wykonywane asynchronicznie po powrocie funkcji . Problem polega zatem na tym, jak i gdzie przechowywać wyniki pobrane przez wywołanie asynchroniczne po jego zwróceniu.
W tym wątku podano kilka rozwiązań. Być może najłatwiejszym sposobem jest przekazanie obiektu do foo()
metody i zapisanie wyników w elemencie tego obiektu po zakończeniu wywołania asynchronicznego.
function foo(result) {
$.ajax({
url: '...',
success: function(response) {
result.response = response; // Store the async result
}
});
}
var result = { response: null }; // Object to hold the async result
foo(result); // Returns before the async completes
Pamiętaj, że wywołanie foo()
nadal nie zwróci nic użytecznego. Jednak wynik wywołania asynchronicznego zostanie teraz zapisany w pliku result.response
.
Użyj callback()
funkcji w foo()
sukcesie. Spróbuj w ten sposób. Jest to proste i łatwe do zrozumienia.
var lat = "";
var lon = "";
function callback(data) {
lat = data.lat;
lon = data.lon;
}
function getLoc() {
var url = "http://ip-api.com/json"
$.getJSON(url, function(data) {
callback(data);
});
}
getLoc();
Korzystanie z Promise
Najdoskonalszą odpowiedzią na to pytanie jest użycie Promise
.
function ajax(method, url, params) {
return new Promise(function(resolve, reject) {
var xhr = new XMLHttpRequest();
xhr.onload = function() {
resolve(this.responseText);
};
xhr.onerror = reject;
xhr.open(method, url);
xhr.send(params);
});
}
Stosowanie
ajax("GET", "/test", "acrive=1").then(function(result) {
// Code depending on result
})
.catch(function() {
// An error occurred
});
Ale poczekaj...!
Jest problem z wykorzystaniem obietnic!
Dlaczego powinniśmy korzystać z naszej własnej, niestandardowej obietnicy?
Używałem tego rozwiązania przez jakiś czas, dopóki nie zorientowałem się, że w starych przeglądarkach jest błąd:
Uncaught ReferenceError: Promise is not defined
Dlatego zdecydowałem się zaimplementować własną klasę Promise dla ES3 do poniższych kompilatorów js, jeśli nie została zdefiniowana. Po prostu dodaj ten kod przed głównym kodem, a następnie bezpiecznie używaj Promise!
if(typeof Promise === "undefined"){
function _classCallCheck(instance, Constructor) {
if (!(instance instanceof Constructor)) {
throw new TypeError("Cannot call a class as a function");
}
}
var Promise = function () {
function Promise(main) {
var _this = this;
_classCallCheck(this, Promise);
this.value = undefined;
this.callbacks = [];
var resolve = function resolve(resolveValue) {
_this.value = resolveValue;
_this.triggerCallbacks();
};
var reject = function reject(rejectValue) {
_this.value = rejectValue;
_this.triggerCallbacks();
};
main(resolve, reject);
}
Promise.prototype.then = function then(cb) {
var _this2 = this;
var next = new Promise(function (resolve) {
_this2.callbacks.push(function (x) {
return resolve(cb(x));
});
});
return next;
};
Promise.prototype.catch = function catch_(cb) {
var _this2 = this;
var next = new Promise(function (reject) {
_this2.callbacks.push(function (x) {
return reject(cb(x));
});
});
return next;
};
Promise.prototype.triggerCallbacks = function triggerCallbacks() {
var _this3 = this;
this.callbacks.forEach(function (cb) {
cb(_this3.value);
});
};
return Promise;
}();
}
Pytanie brzmiało:
Jak zwrócić odpowiedź z wywołania asynchronicznego?
co MOŻE być interpretowane jako:
Jak sprawić, by kod asynchroniczny wyglądał na synchroniczny ?
Rozwiązaniem będzie uniknięcie wywołań zwrotnych i użycie kombinacji Obietnic i asynchronizacji / await .
Chciałbym podać przykład żądania Ajax.
(Chociaż można to napisać w Javascript, wolę pisać w Pythonie i skompilować do Javascript za pomocą Transcrypt . Będzie to wystarczająco jasne).
Najpierw włączmy użycie JQuery, aby było $
dostępne jako S
:
__pragma__ ('alias', 'S', '$')
Zdefiniuj funkcję, która zwraca Obietnicę , w tym przypadku wywołanie Ajax:
def read(url: str):
deferred = S.Deferred()
S.ajax({'type': "POST", 'url': url, 'data': { },
'success': lambda d: deferred.resolve(d),
'error': lambda e: deferred.reject(e)
})
return deferred.promise()
Użyj kodu asynchronicznego tak, jakby był synchroniczny :
async def readALot():
try:
result1 = await read("url_1")
result2 = await read("url_2")
except Exception:
console.warn("Reading a lot failed")
Oczywiście istnieje wiele podejść, takich jak synchroniczne żądanie, obietnica, ale z mojego doświadczenia wynika, że należy użyć metody wywołania zwrotnego. Asynchroniczne zachowanie JavaScript jest naturalne. Tak więc fragment kodu można przepisać nieco inaczej:
function foo() {
var result;
$.ajax({
url: '...',
success: function(response) {
myCallback(response);
}
});
return result;
}
function myCallback(response) {
// Does something.
}
Po przeczytaniu wszystkich odpowiedzi tutaj i na podstawie moich doświadczeń, chciałbym powrócić do szczegółów callback, promise and async/await
dotyczących programowania asynchronicznego w JavaScript.
1) Callback: Podstawowym powodem wywołania zwrotnego jest uruchomienie kodu w odpowiedzi na zdarzenie (patrz przykład poniżej). Za każdym razem używamy wywołania zwrotnego w JavaScript.
const body = document.getElementsByTagName('body')[0];
function callback() {
console.log('Hello');
}
body.addEventListener('click', callback);
Ale jeśli musisz użyć wielu zagnieżdżonych wywołań zwrotnych w poniższym przykładzie, refaktoryzacja kodu będzie straszna.
asyncCallOne(function callback1() {
asyncCallTwo(function callback2() {
asyncCallThree(function callback3() {
...
})
})
})
2) Obietnica: składnia ES6 - Promise rozwiązuje problem piekła zwrotnego!
const myFirstPromise = new Promise((resolve, reject) => {
// We call resolve(...) when what we were doing asynchronously was successful, and reject(...) when it failed.
// In this example, we use setTimeout(...) to simulate async code.
// In reality, you will probably be using something like XHR request or an HTML5 API.
setTimeout(() => {
resolve("Success!") // Yay! Everything went well!
}, 250)
})
myFirstPromise
.then((res) => {
return res.json();
})
.then((data) => {
console.log(data);
})
.catch((e) => {
console.log(e);
});
myFirstPromise to instancja Promise, która reprezentuje proces kodów asynchronicznych. Funkcja rozstrzygania sygnalizuje zakończenie działania instancji Promise. Następnie możemy wywołać .then () (łańcuch .then, jak chcesz) i .catch () na instancji obietnicy:
then — Runs a callback you pass to it when the promise has fulfilled.
catch — Runs a callback you pass to it when something went wrong.
3) Async / Await: nowa składnia ES6 - Await to w zasadzie cukrowa składnia Promise!
Funkcja asynchroniczna zapewnia nam czystą i zwięzłą składnię, która pozwala nam napisać mniej kodu, aby osiągnąć ten sam wynik, jaki uzyskalibyśmy dzięki obietnicom. Async / Await wygląda podobnie do kodu synchronicznego , a kod synchroniczny jest znacznie łatwiejszy do czytania i pisania. Aby wychwycić błędy za pomocą Async / Await, możemy użyć bloku try...catch
. Tutaj nie musisz pisać łańcucha .then () składni Promise.
const getExchangeRate = async () => {
try {
const res = await fetch('https://getExchangeRateData');
const data = await res.json();
console.log(data);
} catch (err) {
console.error(err);
}
}
getExchangeRate();
Wniosek: są to całkowicie trzy składnie programowania asynchronicznego w JavaScript, które powinieneś dobrze zrozumieć. Więc jeśli to możliwe, radzę używać "obietnica" lub "async / await" do refaktoryzacji kodów asynchronicznych (głównie dla żądań XHR) !
Zamiast rzucać w ciebie kodem, istnieją 2 koncepcje, które są kluczem do zrozumienia, jak JS obsługuje wywołania zwrotne i asynchroniczność. (czy to choćby słowo?)
Model pętli zdarzeń i współbieżności
Są trzy rzeczy, o których musisz wiedzieć; Kolejka; pętla zdarzeń i stos
Mówiąc ogólnie, pętla zdarzeń jest jak menedżer projektu, nieustannie nasłuchuje wszelkich funkcji, które chcą uruchomić, i komunikuje się między kolejką a stosem.
while (queue.waitForMessage()) {
queue.processNextMessage();
}
Gdy otrzyma komunikat, aby coś uruchomić, dodaje go do kolejki. Kolejka to lista rzeczy, które czekają na wykonanie (np. Żądanie AJAX). wyobraź sobie to tak:
1. call foo.com/api/bar using foobarFunc
2. Go perform an infinite loop
... and so on
Kiedy jedna z tych wiadomości ma zostać wykonana, zdejmuje wiadomość z kolejki i tworzy stos, stos jest wszystkim, co JS musi wykonać, aby wykonać instrukcję zawartą w wiadomości. Więc w naszym przykładzie mówi się, żeby zadzwonićfoobarFunc
function foobarFunc (var) {
console.log(anotherFunction(var));
}
Więc wszystko, co foobarFunc musi wykonać (w naszym przypadku anotherFunction
), zostanie wrzucone na stos. wykonane, a potem zapomniane - pętla zdarzeń przejdzie do następnej rzeczy w kolejce (lub nasłuchuje komunikatów)
Kluczowa jest tutaj kolejność wykonywania. To jest
KIEDY coś będzie działać
Kiedy wykonujesz połączenie za pomocą AJAX do strony zewnętrznej lub uruchamiasz dowolny kod asynchroniczny (na przykład setTimeout), JavaScript jest zależny od odpowiedzi, zanim będzie można kontynuować.
Najważniejsze pytanie brzmi, kiedy otrzyma odpowiedź? Odpowiedź brzmi: nie wiemy - więc pętla zdarzeń czeka, aż ta wiadomość powie „hej, uruchom mnie”. Gdyby JS po prostu czekał na tę wiadomość synchronicznie, twoja aplikacja zawiesiłaby się i będzie do niczego. Zatem JS kontynuuje wykonywanie następnej pozycji w kolejce, czekając, aż wiadomość zostanie ponownie dodana do kolejki.
Dlatego przy asynchronicznej funkcjonalności używamy rzeczy zwanych callbackami . To trochę jak obietnica całkiem dosłownie. Ponieważ obiecuję, że w pewnym momencie coś zwróci, jQuery używa określonych wywołań zwrotnych, tzw. deffered.done
deffered.fail
I deffered.always
(między innymi). Możesz je wszystkie zobaczyć tutaj
Więc to, co musisz zrobić, to przekazać funkcję, która ma zostać wykonana w pewnym momencie z przekazanymi do niej danymi.
Ponieważ wywołanie zwrotne nie jest wykonywane natychmiast, ale w późniejszym czasie ważne jest, aby przekazać odwołanie do funkcji, która nie została wykonana. więc
function foo(bla) {
console.log(bla)
}
więc przez większość czasu (ale nie zawsze) będziesz przechodzić foo
niefoo()
Mam nadzieję, że to będzie miało jakiś sens. Kiedy napotkasz takie rzeczy, które wydają się zagmatwane - gorąco polecam pełne przeczytanie dokumentacji, aby przynajmniej ją zrozumieć. Dzięki temu będziesz znacznie lepszym programistą.