3.8 KiB
3.8 KiB
id, title, category, status, source_trust_level, verification_status, created_at, updated_at, tags, tech_stack, applied_in, aliases
| id | title | category | status | source_trust_level | verification_status | created_at | updated_at | tags | tech_stack | applied_in | aliases | ||||||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| db-n-plus-one | N+1 쿼리 문제 — 감지와 해결 | Coding | draft | B | conceptual | 2026-05-09 | 2026-05-09 |
|
|
|
N+1 쿼리 문제
1번 쿼리로 N건 가져오고, 각 건마다 추가 쿼리 N번 = N+1. ORM 의 lazy loading 디폴트가 주범. 감지(쿼리 로깅) → JOIN / IN / DataLoader 3가지 해결책.
📖 핵심 개념
N+1 발생 패턴:
const users = await db.users.findMany(); // 1 query → N rows
for (const user of users) {
user.orders = await db.orders.findMany({ where: { userId: user.id } }); // N queries
}
// 총 N+1 queries
해결:
- JOIN (Prisma
include, TypeORMleftJoinAndSelect) - IN list (
WHERE userId IN (...)한 번에) - DataLoader (request-scoped batch + cache)
💻 코드 패턴
1. Prisma include
const users = await prisma.user.findMany({
include: { orders: true }, // JOIN 또는 별도 IN — Prisma 자동
});
// 2 queries 총
2. 직접 batch (raw SQL or query builder)
const users = await db.users.findMany({ where: { active: true } });
const userIds = users.map(u => u.id);
const orders = await db.orders.findMany({ where: { userId: { in: userIds } } });
// 그룹화
const ordersByUser = new Map<string, Order[]>();
for (const o of orders) {
if (!ordersByUser.has(o.userId)) ordersByUser.set(o.userId, []);
ordersByUser.get(o.userId)!.push(o);
}
const result = users.map(u => ({ ...u, orders: ordersByUser.get(u.id) ?? [] }));
3. DataLoader — GraphQL / 복잡 그래프
import DataLoader from 'dataloader';
const orderLoader = new DataLoader(async (userIds: readonly string[]) => {
const orders = await db.orders.findMany({ where: { userId: { in: [...userIds] } } });
const map = groupBy(orders, o => o.userId);
return userIds.map(id => map.get(id) ?? []);
});
// 어디서나
const myOrders = await orderLoader.load(user.id);
// 같은 tick 의 모든 .load 호출이 한 batch 로 합쳐짐
GraphQL resolver 안에서 매 user.orders 가 호출되어도 1 query 로 통합.
4. 감지 — 쿼리 카운트
// dev 미들웨어 — 한 요청 당 쿼리 수 로깅
let count = 0;
prisma.$on('query', () => count++);
app.use((req, res, next) => {
count = 0;
res.on('finish', () => {
if (count > 20) log.warn('high query count', { url: req.url, count });
});
next();
});
🤔 의사결정 기준
| 상황 | 해결 |
|---|---|
| 단순 1:N (User → Orders) | include / IN batch |
| 깊은 그래프 (User → Posts → Comments → Author) | DataLoader |
| GraphQL | DataLoader 거의 필수 |
| 매우 큰 IN (>10k items) | window / chunk 또는 join |
| read replica 활용 | batched read |
❌ 안티패턴
- for loop 안에서 await db.findOne: 정확히 N+1 패턴.
- Promise.all([...users.map(u => fetchOrders(u.id))]): 동시 N 쿼리. 빠르지만 DB 부하.
- JOIN 으로 모두 해결 시도: Cartesian product. User 1 + Orders 100 + Items 1000 = 100,000 행 반환.
- Prisma include 무한 nested: 응답 크기 폭발. 필요한 것만.
- DataLoader 인스턴스 공유: cache 가 영구 → stale. request-scoped (req 마다 새로).
- DataLoader 가 빈 결과를 undefined: 항상 input 길이와 같은 array 반환. null 또는 빈 array.
- 쿼리 로그 미설정: 발견조차 못 함. dev 항상 on.
🤖 LLM 활용 힌트
- "loop 안 await db query 금지. include / IN / DataLoader 중 선택" 강제.
- GraphQL = DataLoader request-scoped 패턴.