MySQL中的事务和锁简单测试

一直以来,对于MySQL中的事务和锁的内容是浅尝辄止,没有花时间了解过,在一次看同事排查的故障中有个问题引起了我的兴趣,虽然过去了很久,但是现在简单总结一下还是有一些收获。
首先我们初始化数据,事务的隔离级别还是MySQL默认的RR,存储引擎为InnoDB
>create table test(id int,name varchar(30));
>insert into test values(1,'aa');
开启一个会话,开启事务。
会话1:
[test]>start transaction;

这个时候我们查看show processlist的信息是不会看到更为具体的SQL等的信息。

我们在另外一个会话中查看事务相关的一个表,Innodb_trx,其实它对应的存储引擎是MEMORY
[information_schema]>select *from innodb_trx\G

然后在会话1执行一条语句。
select * from test where id=1 for update;
再次查看事务表的信息,我们对比前后两次的结果变化,发现唯一的不同是trx_lock_structs的地方,由0变为了2

对于这个字段的含义,可以参考官方文档的介绍。
https://dev.mysql.com/doc/refman/5.6/en/innodb-trx-table.html

对于字段TRX_LOCK_STRUCTS的官方解释如下:
The number of locks reserved by the transaction.

2:
这个时候在会话2中执行语句会发生阻塞,因为存在相应的锁等待。
select * from test where id=1 for update;
等待一段时间,会话2就会提示超时。
[test]>select * from test where id=1 for update;
ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction
这个地方和一个参数是有关联的,innodb_lock_wait_timeout它会控制阻塞等待的时长。
[test]>show variables like '%innodb_lock_wait_timeout%';
| Variable_name            | Value |
| innodb_lock_wait_timeout | 120   |
对于事务相关的信息查看,在MySQL中有三个比较经典的数据字典,innodb_lock_waits,innodb_trx,innodb_trx,三者可以结合起来,就能够查到相对比较完整的阻塞信息和事务的情况,官方提供的一个SQL如下:

我们简称为check_trx.sql,在这个场景中我们运行check_trx.sql会发现线程3573在等待,阻塞它的正是线程3574



这个时候有一个地方需要注意,那就是通过show engine innodb status得到的结果中,标红的部分可以看出锁是表级锁。这个还是和表的结构有一定的关系。
我们可以换一个方式来测试完善,比如测试一下死锁。

测试死锁
首先给表test添加一条记录
insert into test values(2,'bb');
为了杜绝表级锁,对表test 添加主键,如果采用下面的方式添加主键,竟然不可以,看来Oracle用惯了,很多思维方式要复制过来,SQL语法还是有不少地方需要注意。
[test]>alter table test modify id primary key;
ERROR 1064 (42000): You have an error in your SQL syntax; check the manual that corresponds to your MySQL server vline 1。。。
可以使用下面的方式来添加主键。
[test]>ALTER TABLE test ADD UNIQUE INDEX (id), ADD PRIMARY KEY (id);
Query OK, 2 rows affected (0.25 sec)
Records: 2  Duplicates: 0  Warnings: 0
接下来来复现一下死锁的情况。

1
开启事务,更新id=1的那行数据。
start transaction;
[test]>select * from test where id=1 for update;
+----+------+
| id | name |
+----+------+
|  1 | aa   |
+----+------+
1 row in set (0.00 sec)
这个时候查看innodb_trx的信息,只有1条记录。

会话2
开启事务,更新id=2的那行数据。
start transaction;
select * from test where id=2 for update;
(root:localhost:Sat Oct  8 18:15:10 2016)[test]>select * from test where id=2 for update;
+----+------+
| id | name |
+----+------+
|  2 | bb   |
+----+------+
1 row in set (0.00 sec)
这个时候两者是不存在阻塞的情况,因为彼此都是影响独立的行。
>source check_trx.sql
Empty set (0.00 sec)
查看事务表,里面就是2条记录了。


会话1:
在会话1中修改id=2的数据行。
select * from test where id=2 for update;
查看事务表,会有一条阻塞的信息。

会话2
在会话2中修改id=1的数据行,这个时候会发现存在死锁,而MySQL会毫不犹豫的清理掉阻塞的那个会话。这个过程是自动完成的。
[test]>select * from test where id=1 for update;
ERROR 1213 (40001): Deadlock found when trying to get lock; try restarting transaction
查看阻塞的信息,就会发现已经被清理掉了。
[(none)]>source check_trx.sql
Empty set (0.00 sec)
查看事务表,会发现只有1条记录了。

总体感觉MySQL的数据字典还是比较少,不过使用起来还是比较清晰。

时间: 2024-10-30 09:30:11

MySQL中的事务和锁简单测试的相关文章

MySQL中的事务和锁简单用法测试

一直以来,对于MySQL中的事务和锁的内容是浅尝辄止,没有花时间了解过,在一次看同事排查的故障中有个问题引起了我的兴趣,虽然过去了很久,但是现在简单总结一下还是有一些收获. 首先我们初始化数据,事务的隔离级别还是MySQL默认的RR,存储引擎为InnoDB > create table test(id int,name varchar(30)); > insert into test values(1,'aa'); 开启一个会话,开启事务.会话1: [test]>start transa

简单介绍MySQL中的事务机制

  从一个问题开始 最近银行这个事情闹的比较厉害啊,很多储户的钱放在银行,就不翼而飞了,而银行还不管不问,说是用户的责任,打官司,用户还能输了,这就是"社会主义".咱还是少发牢骚,多种树,莫谈国事. 说到银行存钱,就不得不说一下从银行取钱这件事情,从ATM机取钱这件简单的事情,实际上主要分为以下几个步骤: 登陆ATM机,输入密码; 连接数据库,验证密码; 验证成功,获得用户信息,比如存款余额等; 用户输入需要取款的金额,按下确认键; 从后台数据库中减掉用户账户上的对应金额; ATM吐出

简单介绍MySQL中的事务机制_Mysql

从一个问题开始 最近银行这个事情闹的比较厉害啊,很多储户的钱放在银行,就不翼而飞了,而银行还不管不问,说是用户的责任,打官司,用户还能输了,这就是"社会主义".咱还是少发牢骚,多种树,莫谈国事. 说到银行存钱,就不得不说一下从银行取钱这件事情,从ATM机取钱这件简单的事情,实际上主要分为以下几个步骤:     登陆ATM机,输入密码:     连接数据库,验证密码:     验证成功,获得用户信息,比如存款余额等:     用户输入需要取款的金额,按下确认键:     从后台数据库中减

大众点评工程师:从特性说起,漫谈MySQL中的事务及其实现

最近一直在做订单类的项目,使用了事务.我们的数据库选用的是MySQL,存储引擎选用innoDB,innoDB对事务有着良好的支持.这篇文章我们一起来扒一扒事务相关的知识. 为什么要有事务? 事务广泛的运用于订单系统.银行系统等多种场景.如果有以下一个场景:A用户和B用户是银行的储户.现在A要给B转账500元.那么需要做以下几件事: 1. 检查A的账户余额>500元: 2. A账户扣除500元: 3. B账户增加500元: 正常的流程走下来,A账户扣了500,B账户加了500,皆大欢喜.那如果A账

全面了解MySql中的事务_Mysql

最近一直在做订单类的项目,使用了事务.我们的数据库选用的是MySql,存储引擎选用innoDB,innoDB对事务有着良好的支持.这篇文章我们一起来扒一扒事务相关的知识. 为什么要有事务? 事务广泛的运用于订单系统.银行系统等多种场景.如果有以下一个场景:A用户和B用户是银行的储户.现在A要给B转账500元.那么需要做以下几件事: 1. 检查A的账户余额>500元: 2. A账户扣除500元: 3. B账户增加500元: 正常的流程走下来,A账户扣了500,B账户加了500,皆大欢喜.那如果A账

MySQL中的行级锁、表级锁、页级锁_Mysql

在计算机科学中,锁是在执行多线程时用于强行限制资源访问的同步机制,即用于在并发控制中保证对互斥要求的满足. 在DBMS中,可以按照锁的粒度把数据库锁分为行级锁(INNODB引擎).表级锁(MYISAM引擎)和页级锁(BDB引擎 ). 一.行级锁 行级锁是Mysql中锁定粒度最细的一种锁,表示只针对当前操作的行进行加锁.行级锁能大大减少数据库操作的冲突.其加锁粒度最小,但加锁的开销也最大.行级锁分为共享锁 和 排他锁. 特点 开销大,加锁慢:会出现死锁:锁定粒度最小,发生锁冲突的概率最低,并发度也

MySQL中的行级锁,表级锁,页级锁

在计算机科学中,锁是在执行多线程时用于强行限制资源访问的同步机制,即用于在并发控制中保证对互斥要求的满足. 在数据库的DBMS中,可以按照锁的粒度把数据库锁分为行级锁(INNODB引擎).表级锁(MYISAM引擎)和页级锁(BDB引擎 ). 行级锁 行级锁是Mysql中锁定粒度最细的一种锁,表示只针对当前操作的行进行加锁.行级锁能大大减少数据库操作的冲突.其加锁粒度最小,但加锁的开销也最大.行级锁分为共享锁和排它锁(MySQL中的共享锁与排他锁) 特点 开销大,加锁慢:会出现死锁:锁定粒度最小,

SQLServer中的事务和锁

  SQL Server中的事务和锁 事务全部是关于原子性(atomicity)的.原子性的概念是指可以把一些事情当作一个单元来看待.从数据库的角度看,它是指应全部执行或全部不执行的一条或多条语句的最小组合. 在事务中可以标记的点: 1. Begin Tran:设置起始点. 2. Commit Tran:使事务成为数据库永久的.不可逆转的一部分. 3. Rollback Tran:本质上说想要忘记它曾经发生过,即使回滚,回到事务发生之初. 4. Save Tran:创建一个特定标识符,只运行作部

讲解MySQL中的事务特性_Mysql

一个事务是一个连续的一组数据库操作,就好像它是一个单一的工作单元进行.换言之,永远不会是完整的事务,除非该组内的每个单独的操作是成功的.如果在事务的任何操作失败,则整个事务将失败. 实际上,会俱乐部许多SQL查询到一个组中,将执行所有的人都一起作为事务的一部分. 事务的特性: 事务有以下四个标准属性的缩写ACID,通常被称为:     原子性: 确保工作单元内的所有操作都成功完成,否则事务将被中止在故障点,和以前的操作将回滚到以前的状态.     一致性: 确保数据库正确地改变状态后,成功提交的