PurpleSchool — курсы программирования онлайн
  • Пути
    • Frontend React разработчик
    • Frontend Vue разработчик
    • Backend разработчик Node.js
    • Fullstack разработчик React / Node.js
    • Mobile разработчик React Native
    • Backend разработчик Golang
    • Devops инженер
    • Backend разработчик Python
  • AI для кодаНовое
  • О нас
    • Отзывы
    • Реферальная программа
    • О компании
    • Контакты
  • Иконка открытия меню
    • Сообщество
    • PurpleПлюс
    • AI Собеседование
    • AI тренажёр
    • Проекты
PurpleSchool — платформа бесплатных roadmap и курсов для разработчиков
ютуб иконка
Telegram иконка
VK иконка
VK иконка
Курсы
ГлавнаяКаталог курсовFrontendBackendFullstack
Практика
КарьераПроектыPurpleПлюс
Материалы
БлогБаза знаний
Документы
Договор офертаПолитика конфиденциальностиПроверка сертификатаМиграция курсовРеферальная программа
Реквизиты
ИП Ларичев Антон АндреевичИНН 773373765379contact@purpleschool.ru

PurpleSchool © 2020 -2026 Все права защищены

  • Курсы
    • FrontendИконка стрелки
    • AI разработкаИконка стрелки
    • BackendИконка стрелки
    • DevOpsИконка стрелки
    • MobileИконка стрелки
    • ТестированиеИконка стрелки
    • Soft-skillsИконка стрелки
    • ДизайнИконка стрелки
    Иконка слояПерейти в каталог курсов
  • Бесплатно
    • Курсы
    • JavaScript Основы разработкиPython Основы PythonCSS CSS FlexboxКарта развитияВопросы для собеседований
    • База знанийИконка стрелки
    • Новостные рассылкиИконка стрелки
  • PurpleSchool — курсы программирования онлайн
    • AI для кодаНовое
    • Сообщество
    • PurpleПлюс
    • AI Собеседование
    • AI тренажёр
    • Проекты
    Главная
    Сообщество
    Асинхронный JavaScript: callbacks, Promise, async/await по порядку

    Асинхронный JavaScript: callbacks, Promise, async/await по порядку

    Аватар автора Асинхронный JavaScript: callbacks, Promise, async/await по порядку

    Антон Ларичев

    Иконка календаря19 мая 2026
    JavaScriptАсинхронностьPromiseasync/awaitjuniorИконка уровня junior
    Картинка поста Асинхронный JavaScript: callbacks, Promise, async/await по порядку

    Введение

    Асинхронный JavaScript — это фундамент современной веб-разработки. Браузер и Node.js работают в одном потоке, поэтому без асинхронных механизмов любая долгая операция (запрос к серверу, чтение файла, таймер) блокировала бы интерфейс. В этой статье разберём три подхода к работе с асинхронностью в порядке их появления: callbacks, Promise и async/await. Поймём, зачем каждый из них появился и какие проблемы решал.

    Callbacks: с чего всё началось

    Коллбэк — это функция, которую мы передаём в другую функцию как аргумент, чтобы её вызвали позже, когда асинхронная операция завершится. Это самый старый способ работы с асинхронным кодом в JavaScript.

    // Имитация запроса к серверу с коллбэком
    function loadUser(id, callback) {
      setTimeout(() => {
        // Возвращаем пользователя через 1 секунду
        callback(null, { id, name: 'Анна' });
      }, 1000);
    }
    
    loadUser(1, (error, user) => {
      if (error) {
        console.error('Ошибка загрузки:', error);
        return;
      }
      console.log('Пользователь:', user);
    });
    

    Первым параметром коллбэка по соглашению передают ошибку (error-first callback), вторым — результат. Это позволяет единообразно обрабатывать сбои.

    Callback hell

    Когда нужно выполнить несколько асинхронных операций последовательно, код быстро превращается в «лестницу» из вложенных функций:

    loadUser(1, (err, user) => {
      if (err) return console.error(err);
      loadPosts(user.id, (err, posts) => {
        if (err) return console.error(err);
        loadComments(posts[0].id, (err, comments) => {
          if (err) return console.error(err);
          // Глубина вложенности растёт с каждой операцией
          console.log(comments);
        });
      });
    });
    

    Такой код сложно читать, отлаживать и поддерживать. Именно эта проблема породила следующий подход.

    Promise: контракт на будущее значение

    Promise — это объект, представляющий результат асинхронной операции, которая ещё не завершилась. Появился в стандарте ES2015. У промиса три состояния: pending (ожидание), fulfilled (выполнен) и rejected (отклонён).

    function loadUser(id) {
      // Возвращаем промис вместо приёма коллбэка
      return new Promise((resolve, reject) => {
        setTimeout(() => {
          if (id <= 0) {
            reject(new Error('Неверный идентификатор'));
            return;
          }
          resolve({ id, name: 'Анна' });
        }, 1000);
      });
    }
    
    loadUser(1)
      .then((user) => console.log('Пользователь:', user))
      .catch((error) => console.error('Ошибка:', error));
    

    Главное преимущество — цепочки .then(). Каждый then возвращает новый промис, поэтому асинхронные операции выстраиваются в плоскую последовательность:

    loadUser(1)
      .then((user) => loadPosts(user.id))
      .then((posts) => loadComments(posts[0].id))
      .then((comments) => console.log(comments))
      .catch((error) => console.error('Где-то в цепочке ошибка:', error));
    

    Один .catch() в конце ловит ошибки со всех шагов — это намного удобнее проверки if (err) в каждом коллбэке.

    Параллельный запуск

    Promise.all позволяет дождаться выполнения нескольких независимых операций параллельно:

    // Запускаем три запроса одновременно, ждём все
    Promise.all([loadUser(1), loadUser(2), loadUser(3)])
      .then((users) => console.log('Все загружены:', users))
      .catch((error) => console.error(error));
    

    Есть также Promise.allSettled (ждать все, даже отклонённые), Promise.race (первый завершившийся) и Promise.any (первый успешный).

    async/await: синтаксический сахар над Promise

    Конструкция async/await появилась в ES2017 и сделала асинхронный код визуально похожим на синхронный. Под капотом это всё те же промисы.

    // Функция, помеченная async, всегда возвращает промис
    async function showUserData(id) {
      try {
        const user = await loadUser(id);
        const posts = await loadPosts(user.id);
        const comments = await loadComments(posts[0].id);
        console.log(comments);
      } catch (error) {
        // Ловим ошибки любого из await обычным try/catch
        console.error('Ошибка:', error);
      }
    }
    
    showUserData(1);
    

    Ключевое слово await приостанавливает выполнение функции до завершения промиса и возвращает его результат. Использовать await можно только внутри async-функций или на верхнем уровне модуля.

    Параллельность в async/await

    Частая ошибка — забывать о параллельности. Этот код выполняется последовательно и медленно:

    // Плохо: каждый запрос ждёт предыдущий
    const user1 = await loadUser(1);
    const user2 = await loadUser(2);
    

    Правильно — запустить промисы одновременно и дождаться обоих:

    // Хорошо: запросы идут параллельно
    const [user1, user2] = await Promise.all([loadUser(1), loadUser(2)]);
    

    Частые ошибки

    Забытый return в then. Если внутри .then() не вернуть промис, следующий шаг цепочки не дождётся его завершения и получит undefined.

    Смешивание подходов. Не оборачивайте промисы в коллбэки и не создавайте new Promise вокруг уже существующего промиса — это антипаттерн, известный как «explicit promise construction».

    Отсутствие обработки ошибок. Промис без .catch() или await без try/catch приведёт к unhandled rejection. В современных средах это уже считается ошибкой.

    await в цикле forEach. Метод forEach не понимает асинхронности — используйте обычный for...of, если нужна последовательность, или Promise.all с map, если порядок не важен.

    Игнорирование возвращаемого промиса. Вызов async-функции без await запускает её, но не ждёт результата. Иногда это нужно, но чаще — источник скрытых багов.

    Заключение

    Коллбэки, промисы и async/await — это не конкурирующие технологии, а эволюция одной идеи. Промисы решили проблему callback hell, а async/await сделал работу с промисами максимально близкой к синхронному коду. На практике в современных проектах вы будете писать преимущественно async/await, но понимать промисы под капотом необходимо: без этого сложно отлаживать ошибки и грамотно использовать Promise.all, Promise.race и другие комбинаторы. Коллбэки же по-прежнему встречаются в старых API и низкоуровневых модулях Node.js — знать их полезно, чтобы при необходимости легко обернуть в промис через util.promisify или вручную.

    Иконка глаза249

    Комментарии

    0

    Постройте личный план изучения JavaScript с нуля - основы языка и практика для начинающих до уровня Middle — бесплатно!

    JavaScript с нуля - основы языка и практика для начинающих — часть карты развития Frontend, Backend, Mobile

    • step100+ шагов развития
    • lessons30 бесплатных лекций
    • lessons300 бонусных рублей на счет

    Бесплатные лекции

    Лучшие курсы по теме

    изображение курса

    Продвинутый JavaScript

    Антон Ларичев
    AI-тренажерыAI-тренажеры
    Практика в студииПрактика в студии
    Гарантия
    Бонусы
    иконка звёздочки рейтинга4.8
    3 999 ₽ 6 990 ₽
    Подробнее
    изображение курса

    TypeScript с нуля

    Антон Ларичев
    AI-тренажерыAI-тренажеры
    Практика в студииПрактика в студии
    Гарантия
    Бонусы
    иконка звёздочки рейтинга4.8
    3 999 ₽ 6 990 ₽
    Подробнее
    изображение курса

    Neovim

    Антон Ларичев
    Гарантия
    Бонусы
    иконка звёздочки рейтинга4.8
    3 999 ₽ 6 990 ₽
    Подробнее

    Похожие статьи

    Картинка поста SOLID принципы в JavaScript: разбор на примерах
    Иконка аватараАнтон
    Иконка календаря26 июня 2026
    JavaScriptSOLIDООП+ 3middleИконка уровня middle

    SOLID принципы в JavaScript: разбор на примерах

    SOLID принципы в JavaScript — разбираем каждый принцип на реальных примерах кода, чтобы писать чистую и расширяемую архитектуру.

    Иконка чипа0
    Иконка глаза105
    Иконка комментариев0
    Картинка поста Замыкания в JavaScript: полное руководство с примерами
    Иконка аватараАнтон
    Иконка календаря25 июня 2026
    JavaScriptзамыканияосновы JS+ 2juniorИконка уровня junior

    Замыкания в JavaScript: полное руководство с примерами

    Замыкания в JavaScript — разбираем простыми словами: что захватывает функция, как работает scope chain и где применять на практике.

    Иконка чипа0
    Иконка глаза117
    Иконка комментариев0
    Картинка поста CI/CD с GitHub Actions: автоматизация деплоя для начинающих
    Иконка аватараАнтон
    Иконка календаря03 июля 2026
    CI/CDGitHub ActionsDevOps+ 2juniorИконка уровня junior

    CI/CD с GitHub Actions: автоматизация деплоя для начинающих

    CI/CD с GitHub Actions: пошаговая настройка pipeline для автоматического тестирования и деплоя Node.js-приложений без сторонних сервисов.

    Иконка чипа0
    Иконка глаза16
    Иконка комментариев0
    Иконка чипа0