add brain

This commit is contained in:
2026-03-12 15:17:52 +07:00
parent fd9f558fa1
commit e7821a7a9d
355 changed files with 93784 additions and 24 deletions

View File

@@ -0,0 +1,72 @@
---
title: InnoDB Deadlock Resolution
description: Deadlock diagnosis
tags: mysql, deadlocks, innodb, transactions, locking, concurrency
---
# Deadlocks
InnoDB auto-detects deadlocks and rolls back one transaction (the "victim").
## Common Causes
1. **Opposite row ordering** — Transactions accessing the same rows in different order can deadlock. Fix: always access rows in a consistent order (typically by primary key or a common index) so locks are acquired in the same sequence.
2. **Next-key lock conflicts** (REPEATABLE READ) — InnoDB uses next-key locks (row + gap) to prevent phantoms. Fix: use READ COMMITTED (reduces gap locking) or narrow lock scope.
3. **Missing index on WHERE column** — UPDATE/DELETE without an index may require a full table scan, locking many rows unnecessarily and increasing deadlock risk.
4. **AUTO_INCREMENT lock contention** — Concurrent INSERT patterns can deadlock while contending on the auto-inc lock. Fix: use `innodb_autoinc_lock_mode=2` (interleaved) for better concurrency when safe for your workload, or batch inserts.
Note: SERIALIZABLE also uses gap/next-key locks. READ COMMITTED reduces some gap-lock deadlocks but doesn't eliminate deadlocks from opposite ordering or missing indexes.
## Diagnosing
```sql
-- Last deadlock details
SHOW ENGINE INNODB STATUS\G
-- Look for "LATEST DETECTED DEADLOCK" section
-- Current lock waits (MySQL 8.0+)
SELECT object_name, lock_type, lock_mode, lock_status, lock_data
FROM performance_schema.data_locks WHERE lock_status = 'WAITING';
-- Lock wait relationships (MySQL 8.0+)
SELECT
w.requesting_thread_id,
w.requested_lock_id,
w.blocking_thread_id,
w.blocking_lock_id,
l.lock_type,
l.lock_mode,
l.lock_data
FROM performance_schema.data_lock_waits w
JOIN performance_schema.data_locks l ON w.requested_lock_id = l.lock_id;
```
## Prevention
- Keep transactions short. Do I/O outside transactions.
- Ensure WHERE columns in UPDATE/DELETE are indexed.
- Use `SELECT ... FOR UPDATE` sparingly. Batch large updates with `LIMIT`.
- Access rows in a consistent order (by PK or index) across all transactions.
## Retry Pattern (Error 1213)
In applications, retries are a common workaround for occasional deadlocks.
**Important**: ensure the operation is idempotent (or can be safely retried) before adding automatic retries, especially if there are side effects outside the database.
```pseudocode
def execute_with_retry(db, fn, max_retries=3):
for attempt in range(max_retries):
try:
with db.begin():
return fn()
except OperationalError as e:
if e.args[0] == 1213 and attempt < max_retries - 1:
time.sleep(0.05 * (2 ** attempt))
continue
raise
```
## Common Misconceptions
- **"Deadlocks are bugs"** — deadlocks are a normal part of concurrent systems. The goal is to minimize frequency, not eliminate them entirely.
- **"READ COMMITTED eliminates deadlocks"** — it reduces gap/next-key lock deadlocks, but deadlocks still happen from opposite ordering, missing indexes, and lock contention.
- **"All deadlocks are from gap locks"** — many are caused by opposite row ordering even without gap locks.
- **"Victim selection is random"** — InnoDB generally chooses the transaction with lower rollback cost (fewer rows changed).