Назад к вопросам
Middle
142
questionbank

Что такое eager loading в контексте Ruby on Rails?

Sobes Copilot

Получайте ответы в реальном времени

Ответ от нейросети

sobes.tech AI

Eager loading в Ruby on Rails — это механизм, который позволяет подгрузить связанные объекты вместе с основными, выполнив минимальное количество запросов к базе данных. Он решает проблему N+1 запросов.

Проблема N+1 запросов возникает, когда при выборке списка объектов Rails сначала выполняет один запрос для получения самих объектов, а затем N дополнительных запросов (по одному на каждый объект) для получения связанных данных.

Eager loading позволяет подгрузить все связанные данные за один или несколько дополнительных запросов, а не N.

Основные способы реализации eager loading в Rails:

  1. includes: Самый универсальный способ. Rails умеет автоматически выбирать между использованием LEFT OUTER JOIN и двумя отдельными запросами (по одному для основных объектов и по одному для связанных). Чаще использует LEFT OUTER JOIN при использовании .where() на связанных моделях, иначе два отдельных запроса.

    // Загрузит всех авторов вместе с их книгами
    Author.includes(:books).each do |author|
      author.books.each do |book|
        puts book.title // Не вызовет новый запрос к БД
      end
    end
    
  2. preload: Всегда выполняет отдельные запросы для каждого связанного отношения. Это может быть эффективнее, чем includes с LEFT OUTER JOIN, если нет необходимости фильтровать по связанным моделям.

    // Загрузит все книги, а затем отдельным запросом всех авторов этих книг
    Book.preload(:author).each do |book|
      puts book.author.name // Не вызовет новый запрос к БД
    end
    
  3. eager_load: Всегда использует LEFT OUTER JOIN для подгрузки связанных данных. Полезно, когда нужно отфильтровать или отсортировать по полям связанных моделей.

    // Загрузит авторов и их книги, используя LEFT OUTER JOIN, и отфильтрует по названию книги
    Author.eager_load(:books).where("books.title LIKE ?", "%Ruby%").each do |author|
      puts author.name // Не вызовет новый запрос к БД
    end
    

Выбор между includes, preload и eager_load зависит от конкретной задачи и того, нужно ли фильтровать или сортировать по связанным данным. includes часто является хорошим выбором по умолчанию благодаря своей универсальности.