怎么发现mysql的死锁 查看mysql是否有死锁

MySQL数据库中查询表是否被锁以及解锁1.查看表被锁状态
2.查看造成死锁的sql语句
3.查询进程
4.解锁(删除进程)
5.查看正在锁的事物(8.0以下版本)
6.查看等待锁的事物 (8.0以下版本)
怎么查看mysql表是否被锁定当你开始执行一个 ALTER,而你遇到了可怕的“元数据锁定等待”,我敢肯定你一定遇见过 。我最近遇到了一个案例,其中被更改的表要执行一个很小范围的更新(100行) 。ALTER 在负载测试期间一直等待了几个小时 。在停止负载测试后,ALTER 按预期在不到一秒的时间内就完成了 。那么这里发生了什么?
检查外键
每当有奇数次锁定时 , 我的第一直觉就是检查外键 。当然这张表有一些外键引用了一个更繁忙的表 。但是这种行为似乎仍然很奇怪 。对表运行 ALTER 时,会针对子表请求一个 SHARED_UPGRADEABLE 元数据锁 。还有针对父级的 SHARED_READ_ONLY 元数据锁 。
我们来看看如何根据文档获取元数据锁定[1]:
如果给定锁定有多个服务器 , 则首先满足最高优先级锁定请求,并且与 max_write_lock_count系统变量有关 。写锁定请求的优先级高于读取锁定请求 。
[1]:
请务必注意锁定顺序是序列化的:语句逐个获取元数据锁,而不是同时获?。⒃诖斯讨兄葱兴浪觳?。
通常在考虑队列时考虑先进先出 。如果我发出以下三个语句(按此顺序),它们将按以下顺序完成:
1. INSERT INTO parent2. ALTER TABLE child3. INSERT INTO parent
但是当子 ALTER 语句请求对父进行读取锁定时,尽管排序,但两个插入将在 ALTER 之前完成 。以下是可以演示此示例的示例场景:
数据初始化:
CREATE TABLE `parent` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`val` varchar(10) DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB;
CREATE TABLE `child` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`parent_id` int(11) DEFAULT NULL,
`val` varchar(10) DEFAULT NULL,
PRIMARY KEY (`id`),
KEY `idx_parent` (`parent_id`),
CONSTRAINT `fk_parent` FOREIGN KEY (`parent_id`) REFERENCES `parent` (`id`) ON DELETE CASCADE ON UPDATE NO ACTION
) ENGINE=InnoDB;
INSERT INTO `parent` VALUES (1, "one"), (2, "two"), (3, "three"), (4, "four");
Session 1:
start transaction;update parent set val = "four-new" where id = 4;
Session 2:
alter table child add index `idx_new` (val);
Session 3:
start transaction;update parent set val = "three-new" where id = 3;
此时,会话 1 具有打开的事务,并且处于休眠状态 , 并在父级上授予写入元数据锁定 。会话 2 具有在子级上授予的可升级(写入)锁定,并且正在等待父级的读取锁定 。最后会话 3 具有针对父级的授权写入锁定:
mysql select * from performance_schema.metadata_locks; ------------- ------------- ------------------- --------------- ------------- | OBJECT_TYPE | OBJECT_NAME | LOCK_TYPE| LOCK_DURATION | LOCK_STATUS | ------------- ------------- ------------------- --------------- ------------- | TABLE| child| SHARED_UPGRADABLE | TRANSACTION| GRANTED| - ALTER (S2)| TABLE| parent| SHARED_WRITE| TRANSACTION| GRANTED| - UPDATE (S1)| TABLE| parent| SHARED_WRITE| TRANSACTION| GRANTED| - UPDATE (S3)| TABLE| parent| SHARED_READ_ONLY| STATEMENT| PENDING| - ALTER (S2) ------------- ------------- ------------------- --------------- -------------
请注意,具有挂起锁定状态的唯一会话是会话 2(ALTER) 。会话 1 和会话 3 (分别在 ALTER 之前和之后发布)都被授予了写锁 。排序失败的地方是在会话 1 上发生提交的时候 。在考虑有序队列时,人们会期望会话 2 获得锁定,事情就会继续进行 。但是,由于元数据锁定系统的优先级性质,会话 3 具有锁定,会话 2 仍然等待 。
如果另一个写入会话进入并启动新事务并获取针对父表的写锁定,则即使会话 3 完成,ALTER 仍将被阻止 。
只要我保持一个对父表打开元数据锁定的活动事务 , 子表上的 ALTER 将永远不会完成 。更糟糕的是 , 由于子表上的写锁定成功(但是完整语句正在等待获取父读锁定),所以针对子表的所有传入读取请求都将被阻止!
另外,请考虑一下您通常如何对无法完成的语句进行故障排除 。您查看已经打开较长时间的事务(在进程列表和 InnoDB 状态中) 。但由于阻塞线程现在比 ALTER 线程更年轻,因此您将看到的最旧的事务/线程是 ALTER。
这正是这种情况下发生的情况 。在准备发布时,我们的客户端正在运行 ALTER 语句并结合负载测试(一种非常好的做法?。┮匀繁K忱⒉?。问题是负载测试保持对父表打开一个活动的写事务 。这并不是说它只是一直在写,而是有多个线程,一个总是活跃的 。这阻止了 ALTER 完成并阻止对相对静态的子表的随后的读请求 。
幸运的是,这个问题有一个解决方案(除了从设计模式中驱逐外键) 。变量 max_write_lock_count[2] 可用于允许在写入锁定之后在读取锁定之前授予读取锁定连续写锁 。默认情况下,此变量设置为 18446744073709551615,如果你对该表发出 10,000 次写入/秒 , 那么你的读将被锁定 5800 万年……
如何查看MySQL数据库的死锁信息使用终端或命令提示符登录到MySQL , 输入命令:
mysql -h xxxx.xxx.xxx -P 3306 -u username -p password
2
在MySQL客户端下输入命令:
show engine innodb status \G;
3
在打印出来的信息中找到“LATEST DETECTED DEADLOCK”一节内容
4
分析其中的内容,我们就可以知道最近导致死锁的事务有哪些
mysql 查看死锁第一步怎么发现mysql的死锁,查出已锁的进程
查看正在锁的事务
SELECT * FROM INFORMATION_SCHEMA.INNODB_LOCKS;
``
查看等待锁的事务
SELECT * FROM INFORMATION_SCHEMA.INNODB_LOCK_WAITS;
``
INNODB_TRX表主要是包含怎么发现mysql的死锁了正在InnoDB引擎中执行的所有事务的信息,包括waiting for a lock和running的事务
select * from information_schema.innodb_trx
``
第二步 , kill进程
【怎么发现mysql的死锁 查看mysql是否有死锁】 show engin innodb status;//最后一次死锁信息及sql
show open tables where in_use0 //查看锁表
如何查mysql死锁进程查询死锁进程
采用如下存储过程来查询数据中当前造成死锁的进程 。
drop procedure sp_who_lock
go
CREATE procedure sp_who_lock
as
begin
declare @spid int
declare @blk int
declare @count int
declare @index int
declare @lock tinyint
set @lock=0
create table #temp_who_lock
(
id int identity(1,1),
spid int,
blk int
)
if @@error0 return @@error
insert into #temp_who_lock(spid,blk)
select 0 ,blocked
from (select * from master..sysprocesses where blocked0)a
where not exists(select * frommaster..sysprocesses where a.blocked =spid and blocked0)
union select spid,blocked frommaster..sysprocesses where blocked0
if @@error0 return @@error
select @count=count(*),@index=1 from #temp_who_lock
if @@error0 return @@error
if @count=0
begin
select '没有阻塞和死锁信息'
return 0
end
while @indexA href="mailto:=@count"=@count
begin
if exists(select 1 from #temp_who_lock a where id@index and exists(select 1 from #temp_who_lock where idA href="mailto:=@index"=@index and a.blk=spid))
begin
set @lock=1
select @spid=spid,@blk=blk from #temp_who_lock where id=@index
select '引起数据库死锁的是: 'CAST(@spid AS VARCHAR(10))'进程号,其执行的SQL语法如下'
select@spid, @blk
dbcc inputbuffer(@spid)
dbcc inputbuffer(@blk)
end
set @index=@index 1
end
if @lock=0
begin
set @index=1
while @indexA href="mailto:=@count"=@count
begin
select @spid=spid,@blk=blk from #temp_who_lock where id=@index
if @spid=0
select '引起阻塞的是:' cast(@blk as varchar(10))'进程号,其执行的SQL语法如下'
else
select '进程号SPID:'CAST(@spid AS VARCHAR(10))'被''进程号SPID:'CAST(@blk AS VARCHAR(10))'阻塞,其当前进程执行的SQL语法如下'
dbcc inputbuffer(@spid)
dbcc inputbuffer(@blk)
set @index=@index 1
end
end
drop table #temp_who_lock
return 0
end
GO
--执行该存储过程
exec sp_who_lock
补充:
一、产生死锁的原因
在SQL Server中怎么发现mysql的死锁 , 阻塞更多的是产生于实现并发之间的隔离性 。为了使得并发连接所做的操作之间的影响到达某一期望值而对资源人为的进行加锁(锁本质其实可以看作是一个标志位) 。当一个连接对特定的资源进行操作时怎么发现mysql的死锁,另一个连接同时对同样的资源进行操作就会被阻塞怎么发现mysql的死锁,阻塞是死锁产生的必要条件 。
二、如何避免死锁
1.使用事务时,尽量缩短事务的逻辑处理过程,及早提交或回滚事务;
2.设置死锁超时参数为合理范围,如:3分钟-10分种;超过时间 , 自动放弃本次操作,避免进程悬挂;
3.优化程序,检查并避免死锁现象出现;
4.对所有的脚本和SP都要仔细测试,在正是版本之前;
5.所有的SP都要有错误处理(通过@error);
6.一般不要修改SQL SERVER事务的默认级别 。不推荐强行加锁 。
三、处理死锁
1、最简单的处理死锁的方法就是重启服务 。
2、根据指定的死锁进程ID进行处理
根据第二步查询到的死锁进行,大致分析造成死锁的原因 , 并通过如下语句释放该死锁进程
kill pid--pid为查询出来的死锁进程号
3、通过存储过程杀掉某个库下面的所有死锁进程和锁
if exists (select * from dbo.sysobjects where id = object_id(N'[dbo].[sp_killspid]') and OBJECTPROPERTY(id, N'IsProcedure') = 1)
drop procedure [dbo].[sp_killspid]
GO
create proc sp_killspid
@dbname varchar(200)--要关闭进程的数据库名
as
declare @sqlnvarchar(500)
declare @spid nvarchar(20)
declare #tb cursor for
select spid=cast(spid as varchar(20)) from master..sysprocesses where dbid=db_id(@dbname)
open #tb
fetch next from #tb into @spid
while @@fetch_status=0
begin
exec('kill ' @spid)
fetch next from #tb into @spid
end
close #tb
deallocate #tb
go
--使用方法,“db_name”为处理的数据库名称
exec sp_killspid'db_name'
关于怎么发现mysql的死锁和查看mysql是否有死锁的介绍到此就结束了,不知道你从中找到你需要的信息了吗 ?如果你还想了解更多这方面的信息,记得收藏关注本站 。

    推荐阅读