ECMAScript 6 имеет «генераторы», которые позволяют вам легко программировать в асинхронном стиле.
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();
}
}
Для запуска вышеуказанного кода вы делаете это:
const gen = myGenerator(); // Create generator
gen.next(); // Start it
gen.next((...args) => gen.next([...args])); // Set its callback function
Если вам нужно настроить таргетинг на браузеры, которые не поддерживают ES6, вы можете запустить код через Babel или short-compiler для генерации ECMAScript 5.
Обратный вызов ...args
завернут в массив и разрушен, когда вы их читаете так что шаблон может справиться с обратными вызовами, которые имеют несколько аргументов. Например, с узлом fs :
const [err, data] = yield fs.readFile(filePath, "utf-8", callback);
Предположим, что код внутри метода run
изменяет элемент пользовательского интерфейса. Если вы попытаетесь выполнить этот код из потока, отличного от UI, он потерпит неудачу: все операции пользовательского интерфейса должны выполняться в потоке пользовательского интерфейса (также как поток отправки сообщений ).
SwingUtilities.invokeLater
позволяет вам сказать: «Запустите этот бит кода, но делайте это в потоке пользовательского интерфейса». Таким образом, это отлично подходит для фоновых потоков, которые все еще хотят обновить интерфейс. Другой вариант - использовать SwingWorker
, но это не всегда правильно, так как требуется, чтобы код, который «знает», должен использовать поток пользовательского интерфейса, представляет собой код, который устанавливает фоновый поток.
Подробнее см. в учебнике Swing Threading .