MyISAM 只支持表级锁,因其缺乏行级锁所需的元数据结构、事务状态维护、版本号、隐藏列及 Undo Log 等机制,所有写操作均对整表加锁,无法实现行级并发控制。

MyISAM 为什么只支持表级锁?
因为 MyISAM 的设计里没有行级锁所需的元数据结构和并发控制机制。它不维护每行的事务状态、版本号或行锁信息,所有写操作(INSERT、UPDATE、DELETE)都直接对整个表加 WRITE 锁,读操作加 READ 锁 —— 这是引擎层硬编码的行为,没法通过配置开启行锁。
MyISAM 表锁和 InnoDB 行锁的实际表现差异
在高并发更新场景下,这种差异立刻暴露:
-
SELECT COUNT(*) FROM myisam_table会阻塞后续所有INSERT,哪怕只查一行;而 InnoDB 的相同语句通常不锁表(除非用了SELECT ... FOR UPDATE) - 两个连接同时执行
UPDATE myisam_table SET x=1 WHERE id=100和UPDATE myisam_table SET x=2 WHERE id=200,后者必须等前者释放全表锁;InnoDB 中这两条会各自锁定对应行,互不干扰 -
REPAIR TABLE或OPTIMIZE TABLE在 MyISAM 上是独占锁,期间整个表不可读不可写;InnoDB 对应操作(如ALTER TABLE ... ALGORITHM=INPLACE)多数支持在线 DDL
为什么不能给 MyISAM 加行锁?技术上卡在哪
不是“不想”,是根本没实现的基础:
- MyISAM 的索引文件(
.MYI)只存 B-tree 结构和指针,不存事务 ID、回滚指针、隐藏列(如DB_ROW_ID、DB_TRX_ID) - 没有 Undo Log 机制,无法支持 MVCC 或语句级回滚,自然不需要行级锁来协调多版本读写
- 崩溃恢复靠
.MYD文件校验和myisamchk,不依赖 WAL 或 Redo Log,也就没有日志驱动的锁管理上下文
mysql> SHOW CREATE TABLE myisam_example\G
*************************** 1. row ***************************
Table: myisam_example
Create Table: CREATE TABLE `myisam_example` (
`id` int NOT NULL,
`data` varchar(100) DEFAULT NULL
) ENGINE=MyISAM DEFAULT CHARSET=utf8mb4
-- 注意这里:ENGINE=MyISAM,决定了锁粒度上限就是表现在还该用 MyISAM 吗?关键判断点
除非你明确满足以下全部条件,否则不要选:
- 数据完全静态或极少更新(比如配置表、字典表),且
SELECT压力极大 - 服务器内存极小(
- 必须使用全文索引且 MySQL 版本 FULLTEXT)
- 能接受备份时停写、修复时锁表、主从延迟突增等运维代价
现实中,连 ARCHIVE 引擎在只追加场景下都比 MyISAM 更安全——真正绕不开 MyISAM 的情况,现在基本只剩遗留系统迁移前的兼容性锚点。










