解决 MySQL 中的 "Waiting for table metadata lock" 错误
在 MySQL 中,"Waiting for table metadata lock" 错误通常发生在尝试执行 ALTER TABLE 等 DDL 操作时,表的元数据锁被其他事务占用,导致当前操作无法继续。
示例
ALTER TABLE my_table ADD COLUMN new_column INT;
-- Error: Waiting for table metadata lock
常见原因及解决方法
1. 长事务阻塞
长时间运行的事务会阻塞 DDL 操作,导致后续对同一表的操作也被阻塞。
解决方法: 使用 SHOW PROCESSLIST 命令查看正在运行的事务,并终止相关会话。
SHOW PROCESSLIST;
KILL <process_id>;
2. 未提交的事务
未提交的事务会阻塞 DDL 操作,导致表上的锁无法释放。
解决方法: 通过 information_schema.innodb_trx 查看未提交的事务,并终止相关会话。
SELECT * FROM information_schema.innodb_trx \G;
KILL <trx_mysql_thread_id>;
3. 失败的操作
在显式事务中执行失败的操作(如查询不存在的字段)也会导致锁未释放。
解决方法: 通过 performance_schema.events_statements_current 找到失败的语句,并终止相关会话。
SELECT * FROM performance_schema.events_statements_current \G;
KILL <process_id>;
4. 使用 metadata_locks 表
从 MySQL 5.7 开始,可以使用 performance_schema.metadata_locks 表查看等待和持有元数据锁的会话信息。
查询示例:
SELECT
t1.object_schema,
t1.object_name,
t1.lock_type,
t1.lock_status,
t2.PROCESSLIST_ID,
t2.PROCESSLIST_USER,
t2.PROCESSLIST_HOST,
t2.PROCESSLIST_DB,
t2.PROCESSLIST_STATE,
t2.PROCESSLIST_COMMAND,
t2.PROCESSLIST_TIME,
t2.PROCESSLIST_INFO
FROM performance_schema.metadata_locks t1,
performance_schema.threads t2
WHERE t1.owner_thread_id = t2.THREAD_ID
AND t1.object_schema = 'your_database'
AND t1.object_name = 'your_table';
通过这些方法,可以有效解决 "Waiting for table metadata lock" 错误,确保数据库操作顺利进行