DDR爱好者之家 Design By 杰米
基本概念
当前读与快照读
在MVCC中,读操作可以分成两类:快照读 (snapshot read)与当前读 (current read)。 快照读,读取的是记录的可见版本 (有可能是历史版本),不用加锁。当前读,读取的是记录的最新版本,并且对返回的记录,都会加上锁,保证在事务结束前,这条数据都是最新版本。
快照读:简单的select操作,属于快照读,不加锁(Serializable除外)。
select * from table where "htmlcode">select * from table where "color: #ff0000">测试脚本
-- 基本操作 -- -- 查询事务隔离级别,默认是RR show variables like '%isolation%'; -- 设置事务隔离级别为RC set session transaction isolation level read committed; -- 数据初始化 -- begin; drop table if exists user; CREATE TABLE `user` ( `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT, `email` varchar(64) NOT NULL, `age` int(11) NOT NULL, `address` varchar(64) NOT NULL, PRIMARY KEY (`id`), UNIQUE KEY `uniq_email` (`email`), KEY `idx_age` (`age`) ); insert into user (email, age, address) values ("test1@elsef.com", 18, "address1"); insert into user (email, age, address) values ("test2@elsef.com", 20, "address2"); insert into user (email, age, address) values ("test3@elsef.com", 20, "address3"); commit; select * from user; -- 一、trx_id示例 begin; SELECT TRX_ID FROM INFORMATION_SCHEMA.INNODB_TRX WHERE TRX_MYSQL_THREAD_ID = CONNECTION_ID(); select * from user; SELECT TRX_ID FROM INFORMATION_SCHEMA.INNODB_TRX WHERE TRX_MYSQL_THREAD_ID = CONNECTION_ID(); SHOW ENGINE INNODB STATUS; update user set age = 22 where id = 3; -- 查询事务id SELECT TRX_ID FROM INFORMATION_SCHEMA.INNODB_TRX WHERE TRX_MYSQL_THREAD_ID = CONNECTION_ID(); -- INNODB 引擎状态 SHOW ENGINE INNODB STATUS; commit; -- 二、可重复读、不可重复读示例 -- session1 set session transaction isolation level read committed; begin; -- session2 set session transaction isolation level repeatable read; begin; -- session1 select * from user; -- session2 select * from user; -- session3 begin; insert into user (email, age, address) values ("test4@elsef.com", 30, "address4"); commit; -- session1 这里因为是RC,所以可以读到trx3提交的新数据,这里如果是证明不可重复读的话应该使用update而不是insert select * from user; commit; -- session2 这里因为是RR,所以不会读到trx3提交的新数据 select * from user; commit; -- 三、快照读幻读示例 -- session1 set session transaction isolation level repeatable read; begin; -- 这里使用快照读 select * from user; -- session2 begin; insert into user (email, age, address) values ("test4@elsef.com", 30, "address4"); commit; select * from user; -- session1 select * from user; -- 这里读不到test4@的数据,因为是RR -- 这里发生了幻读 insert into user (email, age, address) values ("test4@elsef.com", 30, "address4"); -- 插入失败因为email唯一索引冲突 commit; -- 四、当前读幻读示例 -- RC -- session1 set session transaction isolation level read committed; begin; -- 这里会对所有满足条件的age=20的记录加锁,因为是RC,所以没有GAP锁 delete from user where age = 20; select * from user; -- session2 set session transaction isolation level read committed; begin; -- 因为trx1没有加GAP锁,所以之类可以插入age=20的记录 insert into user (email, age, address) values ("test4@elsef.com", 20, "address4"); select * from user; -- 可以查到4条数据,可以读到trx1的删除数据,因为是RC,trx1未提交所以没影响trx2 commit; -- session1 select * from user; -- 可以读到trx2新插入的数据,虽然trx1是当前读,但是并未添加相应的next-key锁,没有阻止trx2的新数据插入 commit; --RR -- session1 set session transaction isolation level repeatable read; begin; delete from user where age = 20; select * from user; -- session2 begin; -- 这里会阻塞,因为trx1在age=20周围加了GAP锁 -- 非唯一索引,首先,通过索引定位到第一条满足查询条件的记录,加记录上的X锁,加GAP上的GAP锁,然后加主键聚簇索引上的记录X锁; -- 然后读取下一条,重复进行。直至进行到第一条不满足条件的记录,此时,不需要加记录X锁,但是仍旧需要加GAP锁,最后返回结束。 insert into user (email, age, address) values ("test4@elsef.com", 20, "address4"); -- 直到超时,ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction -- 此时如果查询可以看到3条记录 commit; -- session1 -- 此时只能看到1条记录,另外两条被删除了 select * from user; commit; -- 唯一索引+RC -- session1 set session transaction isolation level read committed; begin; delete from user where email = "test3@elsef.com"; -- session2 begin; -- 可以读到,因为trx1是RC select * from user where email = "test3@elsef.com"; -- 尝试更新这个记录的age,会阻塞直到超时,因为email是唯一索引已经被trx1锁住了,同时也会在对应的主键索引上加锁 -- 注意这里操作的id=3就是trx1中操作的email的同一行记录 update user set age = 40 where id = 3; -- session1 commit; -- session2 commit; -- 无索引+RC -- session1 set session transaction isolation level read committed; begin; -- 由于address字段无索引,所以Innodb会对所有行进行加锁,由MySQL server进行判断并释放锁 delete from user where address = "address3"; -- session2 set session transaction isolation level read committed; begin; -- 这一行会成功,因为这一行没有加锁(先加了后释放了) update user set age = 10 where address = "address2"; -- 这一行同样会被阻塞,原因是它已经被trx1的语句加了锁了,全部符合条件的都加锁了 update user set age = 10 where address = "address3"; -- session1 commit; -- session2 commit; -- 非唯一索引+RR -- session1 set session transaction isolation level repeatable read; begin; delete from user where age = 20; -- session2 set session transaction isolation level repeatable read; begin; -- 这里会阻塞,因为trx1中已经锁住了age=20的记录以及加上了GAP锁,所以这里18已经落入锁区间 insert into user (email, age, address) values ("test4@elsef.com", 18, "address4"); -- session1 commit; -- session2 commit; -- 无索引RR -- session1 set session transaction isolation level repeatable read; begin; -- 没有索引,那么会锁上表中的所有记录,同时会锁上主键索引上的所有GAP,杜绝所有的并发更新操作 delete from user where address = "address3"; -- session2 set session transaction isolation level repeatable read; begin; -- 这里会阻塞,原因是主键已经被加上了GAP锁,所以新的插入不能执行成功 insert into user (email, age, address) values ("test4@elsef.com", 18, "address4"); -- session1 commit; -- session2 commit; -- 死锁 简单示例 -- session1 begin; delete from user where id = 1; -- session2 begin; delete from user where id = 3; -- session1 delete from user where id = 3; -- seession2 -- 这里MySQL判断发生了死锁,中断了一个trx -- ERROR 1213 (40001): Deadlock found when trying to get lock; try restarting transaction delete from user where id = 1; -- session1 rollback; -- session2; rollback; -- 五、死锁 insert示例 drop table if exists t1; begin; create table t1 ( `id` bigint not null auto_increment, primary key (`id`) ); insert into t1 values(1); insert into t1 values(5); commit; select * from t1; -- session1 begin; insert into t1 values (2); -- sessioin2 begin; -- 这里会阻塞 insert into t1 values (2); -- session3 begin; -- 这里会阻塞 insert into t1 values (2); -- session1; -- 此时回滚,trx2和trx3收到通知,MySQL自动中断一个trx,因为发生了死锁 -- ERROR 1213 (40001): Deadlock found when trying to get lock; try restarting transaction rollback; --session2; rollback; --session3; rollback;总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。
DDR爱好者之家 Design By 杰米
广告合作:本站广告合作请联系QQ:858582 申请时备注:广告合作(否则不回)
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
DDR爱好者之家 Design By 杰米
暂无评论...
《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线
暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。
艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。
《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。
更新日志
2024年11月24日
2024年11月24日
- 凤飞飞《我们的主题曲》飞跃制作[正版原抓WAV+CUE]
- 刘嘉亮《亮情歌2》[WAV+CUE][1G]
- 红馆40·谭咏麟《歌者恋歌浓情30年演唱会》3CD[低速原抓WAV+CUE][1.8G]
- 刘纬武《睡眠宝宝竖琴童谣 吉卜力工作室 白噪音安抚》[320K/MP3][193.25MB]
- 【轻音乐】曼托凡尼乐团《精选辑》2CD.1998[FLAC+CUE整轨]
- 邝美云《心中有爱》1989年香港DMIJP版1MTO东芝首版[WAV+CUE]
- 群星《情叹-发烧女声DSD》天籁女声发烧碟[WAV+CUE]
- 刘纬武《睡眠宝宝竖琴童谣 吉卜力工作室 白噪音安抚》[FLAC/分轨][748.03MB]
- 理想混蛋《Origin Sessions》[320K/MP3][37.47MB]
- 公馆青少年《我其实一点都不酷》[320K/MP3][78.78MB]
- 群星《情叹-发烧男声DSD》最值得珍藏的完美男声[WAV+CUE]
- 群星《国韵飘香·贵妃醉酒HQCD黑胶王》2CD[WAV]
- 卫兰《DAUGHTER》【低速原抓WAV+CUE】
- 公馆青少年《我其实一点都不酷》[FLAC/分轨][398.22MB]
- ZWEI《迟暮的花 (Explicit)》[320K/MP3][57.16MB]