Jak zwrócić odpowiedź z wywołania asynchronicznego?

Jan 09 2013

Mam funkcję, fooktó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ą undefinedlub jakąkolwiek początkową wartość zmiennej result) .

Przykład użycia ajaxfunkcji 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 thenbloku 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

5905 FelixKling Jan 09 2013 at 00:06

→ 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.$.ajaxreturn 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 findItemmoż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 zasync/await (ES2017 +, dostępne w starszych przeglądarkach, jeśli używasz transpilera lub regeneratora)
  • Wywołania zwrotne (popularne w węźle)
  • Obietnice zthen() (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ą asynci awaitmożesz pisać asynchronicznie w „stylu synchronicznym”. Kod jest nadal asynchroniczny, ale jest łatwiejszy do odczytania / zrozumienia.

async/awaitopiera się na obietnicach: asyncfunkcja 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 awaitwewnątrz asyncfunkcji. Obecnie najwyższy poziom awaitnie jest jeszcze obsługiwany, więc może być konieczne utworzenie asynchronicznego IIFE ( natychmiastowo wywoływanego wyrażenia funkcji ), aby rozpocząć asynckontekst.

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 foozaakceptować oddzwonienie i użyć go jako successoddzwonienia. 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
    });
}

callbackbędzie odnosić się do funkcji, do której przekazujemy, foogdy ją wywołujemy, i do której ją przekazujemy success. To znaczy, gdy żądanie Ajax się powiedzie, $.ajaxwywoła callbacki 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 ifinstrukcja zawsze pobierze ten obiekt odroczony, potraktuje go jako truei 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ż falsejako trzeci argument do .open.

jQuery

Jeśli używasz jQuery , możesz ustawić asyncopcję na false. Zauważ, że ta opcja jest przestarzała od wersji jQuery 1.8. Następnie możesz nadal użyć successwywołania zwrotnego lub uzyskać dostęp do responseTextwł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, $.getJSONitp, 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).

1096 BenjaminGruenbaum May 30 2013 at 06:30

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ść azwracana jest, undefinedponieważ a=5część 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 getFiveakcję 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:

  1. Spraw, aby połączenie AJAX było synchroniczne (nazwijmy to SJAX).
  2. 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 foomoże przyjąć oddzwonienie. Powiemy naszemu kodowi, jak ma reagować po foozakoń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.

411 cocco Aug 19 2013 at 15:06

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:

  1. Jest krótszy niż wszystkie inne wymienione funkcje.
  2. Wywołanie zwrotne jest ustawiane bezpośrednio (więc nie ma dodatkowych niepotrzebnych zamknięć).
  3. Używa nowego onload (więc nie musisz sprawdzać stanu readystate &&)
  4. 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 XMLHttpRequestnazwy 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.statusTextas 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.typeco 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 .txtplik 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.)

326 BenjaminGruenbaum May 12 2015 at 09:22

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 thenobsługi są zawsze wykonywane asynchronicznie - to znaczy po kodzie poniżej nich, którego nie ma w .thenprocedurze obsługi.

Oznacza to, że kiedy zwracasz dataprogram thenobsł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ść datajest, undefinedponieważ data = 5część 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 thenobsługi do obietnic w celu wyodrębnienia ich wartości i obsługi błędów. thenprogramy 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 fetchlub $.getobietnice zwrotu jQuery .

Kiedy zadzwonić .thenna 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ć, thenaby 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.coroutinetutaj używać Bluebird , ale są też inne opakowania, takie jak colub 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 awaitobiecać. To jest po prostu „cukier” (ładniejsza składnia) dla powyższej propozycji ES6 poprzez dodanie słów kluczowych asynci 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 :)

256 Nic May 23 2014 at 09:05

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.

242 HemantBavle Feb 19 2014 at 01:58

Najprostszym rozwiązaniem jest utworzenie funkcji JavaScript i wywołanie jej dla successwywoł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);    
}); 
229 JohannesFahrenkrug Aug 11 2016 at 21:17

Odpowiem okropnie wyglądającym, ręcznie rysowanym komiksem. Drugi obraz jest powodem, dla którego resultznajduje się undefinedw twoim przykładzie kodu.

165 MaleenAbewardana Aug 26 2014 at 15:11

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 Angular2ze spojrzeniem na poniższym przykładzie, ale jego zalecany do stosowania Observablesz 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.

159 T.J.Crowder May 03 2017 at 23:59

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 doSomethingAsyncnie 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ę expectingi po prostu używać results.length === theArray.length, ale to pozostawia nas otwartymi na możliwość theArrayzmiany, gdy połączenia są zaległe ...)

Zwróć uwagę, jak używamy indexfrom, forEachaby zapisać wynik w resultstej 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 Promisezamiast 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 doSomethingAsyncprzekazali nam błędy, rejectodrzucilibyś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, doSomethingAsyncktóre zwraca obietnicę, a następnie wykonaj poniższe czynności ...)

Jeśli doSomethingAsyncdaje 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 doSomethingAsynczignoruje drugi i trzeci argument, możesz po prostu przekazać go bezpośrednio map( mapwywoł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.allwypeł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, doSomethingAsyncktóre daje ci obietnicę i wykonaj poniższe czynności ...)

Jeśli doSomethingAsyncdaje 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;
}

113 FranciscoCarmona Jun 02 2016 at 15:31

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, getJokejest 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);
  });
})();
112 Alireza May 24 2017 at 16:38

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 undefinedna pierwszym etapie, więc masz result = undefinedprzed 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.

105 AnishK. Nov 01 2017 at 03:12

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 resolvewartoś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/catchobietnic 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 promisespod 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);
     }
  }
101 jsbisht Sep 02 2015 at 19:54

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 resultobiektu 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.

89 rohithpr Jan 26 2016 at 00:43

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 namejest 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 .

88 loretoparisi Apr 13 2016 at 05:55

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 XMLHttpRequestobiektu 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 playlistobiektó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 - ExecutionBlockprzeanalizuje wynik, zaplanuje nowy zestaw obietnic w oparciu o tablicę wyników, czyli listę userobiektów Spotify, i ExecutionProfileBlockasynchronicznie 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 searchinterfejsy 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 .

84 PabloMatiasGomez Apr 22 2016 at 21:47

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
});
82 mikemaccana Jun 02 2017 at 16:51

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

80 AniketJha Feb 03 2018 at 13:06

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

70 VinothRajendran May 26 2016 at 20:26

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)
});
70 amaksr May 27 2017 at 09:47

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 datawł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

42 James Feb 17 2018 at 22:26

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 ...argssą 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);
39 MohanDere Aug 13 2016 at 16:36

Oto kilka podejść do pracy z żądaniami asynchronicznymi:

  1. Browser Promise
  2. P: Biblioteka obietnic dla JavaScript
  3. A + Promises.js
  4. jQuery odroczony
  5. Interfejs API XMLHttpRequest
  6. 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();

38 4revs,3users92%user663031 Jan 23 2016 at 10:28

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_milkzakończenie, zanim się wykona put_in_coffee. Innymi słowy, nie wie, że order_milkjest 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_milkuruchamia 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_coffeezaró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_milkzwrot obietnicy za przybycie mleka, a następnie określamy put_in_coffeejako thenczynność 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 returndo wywołania $.ajax. To działa, ponieważ jQuery $.ajaxjuż 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 asynckluczowe

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 ajest 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 - awaitsł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);
}
37 DavidRTribble Sep 24 2015 at 05:52

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.

36 MahfuzurRahman Apr 24 2017 at 15:09

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();
30 AmirFo Dec 07 2018 at 21:10

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;
    }();
}
29 PieterJanBonestroo Jan 14 2018 at 02:13

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")
28 KhoaBui Jul 06 2017 at 03:28

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.
}
26 SanjiMika Jan 20 2020 at 05:23

Po przeczytaniu wszystkich odpowiedzi tutaj i na podstawie moich doświadczeń, chciałbym powrócić do szczegółów callback, promise and async/awaitdotyczą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) !

20 MatthewBrent May 04 2018 at 22:56

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.failI 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ć fooniefoo()

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ą.