Soluções N+1 no Spring Data JPA em 2026: Fetch Join e EntityGraph
Guia completo para detectar e corrigir o problema N+1 no Spring Data JPA. Fetch join, @EntityGraph, batch fetching e estratégias de performance de consultas.

O problema N+1 representa uma das armadilhas de performance mais comuns no JPA. Uma consulta inocente para recuperar 100 pedidos pode disparar 101 consultas SQL: uma para os pedidos, depois uma para cada cliente associado. Essa multiplicação silenciosa de consultas degrada a performance e sobrecarrega o banco de dados.
Um endpoint que retorna 50 artigos com seus autores pode saltar de 10ms para 500ms por causa do N+1. A detecção precoce evita problemas críticos em produção.
Entendendo o Problema N+1 no JPA
O problema N+1 ocorre quando o JPA carrega uma coleção de entidades e depois executa uma consulta adicional para cada entidade carregar suas associações. Esse comportamento decorre do carregamento lazy padrão dos relacionamentos @OneToMany e @ManyToMany.
Considere um modelo clássico com pedidos e clientes. Cada pedido pertence a um cliente, e esse relacionamento usa carregamento lazy por padrão.
@Entity
@Table(name = "orders")
public class Order {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
private String orderNumber;
private LocalDateTime createdAt;
// ManyToOne relationship is lazy by default since JPA 2.0
@ManyToOne(fetch = FetchType.LAZY)
@JoinColumn(name = "customer_id")
private Customer customer;
// OneToMany relationship lazy by default
@OneToMany(mappedBy = "order", fetch = FetchType.LAZY)
private List<OrderItem> items = new ArrayList<>();
// Getters and setters omitted
}@Entity
@Table(name = "customers")
public class Customer {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
private String name;
private String email;
// Getters and setters omitted
}Quando uma consulta recupera pedidos e em seguida acessa o nome do cliente, o Hibernate executa uma consulta adicional para cada pedido.
@Service
@RequiredArgsConstructor
public class OrderService {
private final OrderRepository orderRepository;
public List<OrderDto> getAllOrders() {
// 1 query: SELECT * FROM orders
List<Order> orders = orderRepository.findAll();
// For each order, accessing customer triggers a query
return orders.stream()
.map(order -> new OrderDto(
order.getId(),
order.getOrderNumber(),
// N queries: SELECT * FROM customers WHERE id = ?
order.getCustomer().getName()
))
.toList();
}
}Para 100 pedidos, esse código executa 101 consultas SQL. Os logs do Hibernate revelam esse padrão destrutivo.
-- Query 1: fetch orders
SELECT o.id, o.order_number, o.created_at, o.customer_id FROM orders o
-- Queries 2-101: fetch each customer
SELECT c.id, c.name, c.email FROM customers c WHERE c.id = 1
SELECT c.id, c.name, c.email FROM customers c WHERE c.id = 2
SELECT c.id, c.name, c.email FROM customers c WHERE c.id = 3
-- ... 97 more identical queriesDetectando o Problema N+1 com Logs do Hibernate
O primeiro passo envolve ativar os logs SQL para identificar consultas problemáticas. A configuração a seguir exibe cada consulta executada pelo Hibernate.
# application.yml
spring:
jpa:
show-sql: true
properties:
hibernate:
# Format SQL for better readability
format_sql: true
# Display session statistics (queries, time)
generate_statistics: true
logging:
level:
# Detailed SQL query logging
org.hibernate.SQL: DEBUG
# Display prepared statement parameters
org.hibernate.orm.jdbc.bind: TRACEAs estatísticas do Hibernate fornecem um resumo valioso ao final de cada transação.
Session Metrics {
23421 nanoseconds spent acquiring 1 JDBC connection;
0 nanoseconds spent releasing 0 JDBC connections;
1254789 nanoseconds spent preparing 101 JDBC statements;
15478963 nanoseconds spent executing 101 JDBC statements;
0 nanoseconds spent executing 0 JDBC batches;
}A contagem de 101 statements JDBC para uma simples lista de pedidos sinaliza claramente um problema N+1.
Logs SQL e estatísticas impactam a performance. Essas opções devem permanecer desativadas em produção e reservadas para ambientes de desenvolvimento e teste.
Solução 1: Fetch Join com JPQL
Fetch join carrega as associações em uma única consulta SQL através de um join. Essa abordagem explícita resolve o N+1 recuperando todos os dados necessários de uma só vez.
public interface OrderRepository extends JpaRepository<Order, Long> {
// Explicit fetch join to load customers
@Query("SELECT o FROM Order o JOIN FETCH o.customer")
List<Order> findAllWithCustomer();
// Multiple fetch join for several associations
@Query("SELECT o FROM Order o " +
"JOIN FETCH o.customer c " +
"JOIN FETCH o.items i")
List<Order> findAllWithCustomerAndItems();
// Fetch join with WHERE condition
@Query("SELECT o FROM Order o " +
"JOIN FETCH o.customer c " +
"WHERE o.createdAt > :since")
List<Order> findRecentOrdersWithCustomer(
@Param("since") LocalDateTime since
);
}Fetch join transforma N+1 consultas em uma única consulta otimizada.
-- Single query with join
SELECT o.id, o.order_number, o.created_at, o.customer_id,
c.id, c.name, c.email
FROM orders o
JOIN customers c ON o.customer_id = c.idO serviço agora utiliza o método otimizado sem modificar o código de negócio.
@Service
@RequiredArgsConstructor
public class OrderService {
private final OrderRepository orderRepository;
public List<OrderDto> getAllOrders() {
// Single query with join
List<Order> orders = orderRepository.findAllWithCustomer();
// No additional queries
return orders.stream()
.map(order -> new OrderDto(
order.getId(),
order.getOrderNumber(),
order.getCustomer().getName() // Already loaded
))
.toList();
}
}Pronto para mandar bem nas entrevistas de Spring Boot?
Pratique com nossos simuladores interativos, flashcards e testes tecnicos.
Solução 2: @EntityGraph para Controle Declarativo
A anotação @EntityGraph oferece uma alternativa declarativa ao fetch join. Ela define quais associações carregar sem escrever JPQL personalizado.
public interface OrderRepository extends JpaRepository<Order, Long> {
// Inline EntityGraph with attributePaths
@EntityGraph(attributePaths = {"customer"})
List<Order> findAll();
// EntityGraph with multiple attributes
@EntityGraph(attributePaths = {"customer", "items"})
List<Order> findByCreatedAtAfter(LocalDateTime since);
// Named EntityGraph referencing entity definition
@EntityGraph(value = "Order.withCustomerAndItems")
List<Order> findByCustomerId(Long customerId);
// Combination with custom query
@EntityGraph(attributePaths = {"customer"})
@Query("SELECT o FROM Order o WHERE o.orderNumber LIKE :prefix%")
List<Order> findByOrderNumberPrefix(@Param("prefix") String prefix);
}EntityGraphs nomeados são definidos diretamente na entidade para reúso em vários repositórios.
@Entity
@Table(name = "orders")
@NamedEntityGraph(
name = "Order.withCustomer",
attributeNodes = @NamedAttributeNode("customer")
)
@NamedEntityGraph(
name = "Order.withCustomerAndItems",
attributeNodes = {
@NamedAttributeNode("customer"),
@NamedAttributeNode("items")
}
)
@NamedEntityGraph(
name = "Order.full",
attributeNodes = {
@NamedAttributeNode("customer"),
@NamedAttributeNode(value = "items", subgraph = "items-product")
},
subgraphs = @NamedSubgraph(
name = "items-product",
attributeNodes = @NamedAttributeNode("product")
)
)
public class Order {
// Fields unchanged
}O subgraph permite carregar associações aninhadas. O exemplo acima carrega os pedidos, seus itens e o produto de cada item em uma única consulta.
Solução 3: Batch Fetching para Coleções
Batch fetching oferece uma alternativa ao fetch join para coleções @OneToMany. Em vez de carregar cada coleção individualmente, o Hibernate agrupa as consultas em lotes.
@Entity
@Table(name = "orders")
public class Order {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
private String orderNumber;
@ManyToOne(fetch = FetchType.LAZY)
@JoinColumn(name = "customer_id")
private Customer customer;
// Batch fetching on the collection
@OneToMany(mappedBy = "order")
@BatchSize(size = 25)
private List<OrderItem> items = new ArrayList<>();
}Com @BatchSize(size = 25), o Hibernate carrega itens de 25 pedidos por vez em vez de 1. Para 100 pedidos, a contagem de consultas cai de 101 para 5.
-- Without batch fetching: 100 queries
SELECT * FROM order_items WHERE order_id = 1
SELECT * FROM order_items WHERE order_id = 2
-- ... 98 more queries
-- With @BatchSize(size = 25): 4 queries
SELECT * FROM order_items WHERE order_id IN (1, 2, 3, ..., 25)
SELECT * FROM order_items WHERE order_id IN (26, 27, 28, ..., 50)
SELECT * FROM order_items WHERE order_id IN (51, 52, 53, ..., 75)
SELECT * FROM order_items WHERE order_id IN (76, 77, 78, ..., 100)A configuração global do tamanho de lote se aplica a todas as coleções da aplicação.
# application.yml
spring:
jpa:
properties:
hibernate:
# Global default batch size
default_batch_fetch_size: 25Batch fetching é adequado para casos onde fetch join gera um produto cartesiano excessivamente grande. Para um pedido com 10 itens e 5 pagamentos, fetch join retorna 50 linhas. Batch fetching executa 2 consultas separadas mais eficientes.
Comparando Estratégias de Carregamento
Cada estratégia oferece vantagens dependendo do contexto de uso. A tabela a seguir resume os casos de uso ideais.
| Estratégia | Caso de Uso | Vantagens | Desvantagens |
|----------|----------|------------|---------------|
| Fetch Join | Relacionamentos @ManyToOne | Consulta SQL única | Produto cartesiano com coleções |
| @EntityGraph | Carregamento declarativo | Reutilizável, legível | Menos flexível que JPQL |
| Batch Fetching | Coleções @OneToMany | Evita produto cartesiano | Múltiplas consultas |
| Subselect | Coleções raramente acessadas | Carrega apenas quando necessário | Subconsulta correlacionada |
A estratégia subselect carrega toda a coleção no primeiro acesso a qualquer elemento.
@OneToMany(mappedBy = "order")
@Fetch(FetchMode.SUBSELECT)
private List<OrderItem> items = new ArrayList<>();-- Generated subselect query
SELECT * FROM order_items
WHERE order_id IN (SELECT id FROM orders WHERE created_at > ?)Evitando o N+1 com Projeções DTO
Projeções DTO oferecem uma abordagem radical mas eficaz. Ao selecionar apenas as colunas necessárias, as projeções evitam completamente o carregamento de entidades e suas associações.
public record OrderSummaryDto(
Long orderId,
String orderNumber,
String customerName,
String customerEmail
) {}public interface OrderRepository extends JpaRepository<Order, Long> {
// DTO projection with constructor
@Query("SELECT new com.example.dto.OrderSummaryDto(" +
"o.id, o.orderNumber, c.name, c.email) " +
"FROM Order o JOIN o.customer c")
List<OrderSummaryDto> findAllOrderSummaries();
// DTO projection with condition
@Query("SELECT new com.example.dto.OrderSummaryDto(" +
"o.id, o.orderNumber, c.name, c.email) " +
"FROM Order o JOIN o.customer c " +
"WHERE o.createdAt > :since")
List<OrderSummaryDto> findRecentOrderSummaries(
@Param("since") LocalDateTime since
);
}Essa abordagem gera uma consulta SQL ideal sem o overhead do mapeamento de entidades.
SELECT o.id, o.order_number, c.name, c.email
FROM orders o
JOIN customers c ON o.customer_id = c.id
WHERE o.created_at > ?Configuração Avançada com Spring Data JPA
O Spring Data JPA 3.x introduz melhorias para o gerenciamento de EntityGraph e consultas otimizadas.
public interface OrderRepository extends JpaRepository<Order, Long> {
// Dynamic EntityGraph with Specification
@EntityGraph(attributePaths = {"customer"})
List<Order> findAll(Specification<Order> spec);
// Pagination with EntityGraph
@EntityGraph(attributePaths = {"customer"})
Page<Order> findByCustomerNameContaining(
String name,
Pageable pageable
);
// Slice for efficient pagination
@EntityGraph(attributePaths = {"customer"})
Slice<Order> findByCreatedAtBefore(
LocalDateTime date,
Pageable pageable
);
}O carregamento condicional permite aplicar diferentes estratégias dependendo do contexto.
@Service
@RequiredArgsConstructor
public class OrderService {
private final OrderRepository orderRepository;
private final EntityManager entityManager;
public List<Order> getOrdersWithGraph(String graphName) {
// Dynamic EntityGraph retrieval
EntityGraph<?> graph = entityManager
.getEntityGraph(graphName);
return entityManager
.createQuery("SELECT o FROM Order o", Order.class)
.setHint("jakarta.persistence.loadgraph", graph)
.getResultList();
}
public List<Order> getOrdersForListing() {
// Minimal graph for listing
return getOrdersWithGraph("Order.withCustomer");
}
public List<Order> getOrdersForDetail() {
// Full graph for detail view
return getOrdersWithGraph("Order.full");
}
}Comece a praticar!
Teste seus conhecimentos com nossos simuladores de entrevista e testes tecnicos.
Testes de Performance para Detectar N+1
Testes automatizados verificam a ausência de problemas N+1 contando as consultas SQL executadas.
@DataJpaTest
@AutoConfigureTestDatabase(replace = Replace.NONE)
class OrderRepositoryPerformanceTest {
@Autowired
private OrderRepository orderRepository;
@Autowired
private EntityManager entityManager;
@PersistenceContext
private EntityManager em;
private Statistics statistics;
@BeforeEach
void setUp() {
// Enable Hibernate statistics
Session session = entityManager.unwrap(Session.class);
SessionFactory factory = session.getSessionFactory();
statistics = factory.getStatistics();
statistics.setStatisticsEnabled(true);
statistics.clear();
}
@Test
void findAllWithCustomer_shouldExecuteSingleQuery() {
// Given: 50 orders in database
createTestOrders(50);
entityManager.clear();
statistics.clear();
// When: fetch with fetch join
List<Order> orders = orderRepository.findAllWithCustomer();
// Then: single query executed
assertThat(orders).hasSize(50);
assertThat(statistics.getQueryExecutionCount())
.as("Should execute only 1 query with fetch join")
.isEqualTo(1);
}
@Test
void findAll_withoutOptimization_triggersNPlus1() {
// Given: 50 orders in database
createTestOrders(50);
entityManager.clear();
statistics.clear();
// When: fetch without optimization
List<Order> orders = orderRepository.findAll();
// Access customers to trigger lazy loading
orders.forEach(o -> o.getCustomer().getName());
// Then: N+1 queries (51 instead of 1)
assertThat(statistics.getQueryExecutionCount())
.as("Should detect N+1 problem")
.isGreaterThan(1);
}
private void createTestOrders(int count) {
for (int i = 0; i < count; i++) {
Customer customer = new Customer();
customer.setName("Customer " + i);
customer.setEmail("customer" + i + "@test.com");
entityManager.persist(customer);
Order order = new Order();
order.setOrderNumber("ORD-" + i);
order.setCustomer(customer);
order.setCreatedAt(LocalDateTime.now());
entityManager.persist(order);
}
entityManager.flush();
}
}A asserção sobre getQueryExecutionCount() garante que as otimizações permaneçam em vigor durante a evolução do código.
Conclusão
O problema N+1 representa um grande desafio de performance no Spring Data JPA, mas existem várias soluções eficazes. Fetch join e @EntityGraph resolvem a maioria dos casos carregando as associações em uma única consulta. Batch fetching oferece uma alternativa para grandes coleções onde fetch join gera um produto cartesiano.
Checklist de Prevenção do N+1:
- ✅ Ativar logs SQL em desenvolvimento para detectar consultas múltiplas
- ✅ Usar
JOIN FETCHpara relacionamentos@ManyToOneacessados frequentemente - ✅ Aplicar
@EntityGraphpara carregamento declarativo reutilizável - ✅ Configurar
@BatchSizepara coleções@OneToManygrandes - ✅ Preferir projeções DTO para operações somente leitura
- ✅ Escrever testes que verifiquem o número de consultas executadas
- ✅ Desativar logs SQL e estatísticas em produção
- ✅ Perfilar regularmente endpoints críticos com métricas do Hibernate
Tags
Compartilhar
Artigos relacionados

Entrevista Spring GraphQL: Resolvers, DataLoaders e Soluções para o Problema N+1
Prepare-se para entrevistas Spring GraphQL com este guia completo. Resolvers, DataLoaders, gestão do problema N+1, mutations e melhores práticas para perguntas técnicas.

Spring Boot 3.4 Virtual Threads: Perguntas de Entrevista e Benchmarks de Performance
Domine as Virtual Threads do Java 21 com Spring Boot 3.4: 15 perguntas de entrevista, benchmarks de performance e padrões de migração para vencer entrevistas técnicas.

Logging em Spring Boot 2026: logs estruturados em produção com Logback e JSON
Guia completo de logs estruturados no Spring Boot. Configuração Logback JSON, MDC para tracing, melhores práticas em produção e integração com ELK Stack.