数据库之事务

什么是事务
事务的四大特性(ACID)
事务的并发读问题(脏读,不可重复读,幻读)
事务的隔离级别(读未提交,读已提交,可重复读,串行化)
用cmd演示一下事务隔离级别的效果
jdbc操作事务的隔离级别
注:事务只发生在DML中,和DQL无关!!!

一:什么是事务:

注:在默认情况下,MySQL每执行一条SQL语句,都是一个单独的事务。如果需要在一个事务中包含多条SQL语句,那么需要开启事务和结束事务 

二:事务的四大特性

 三:事务的并发读问题
脏读(dirty read):读到另一个事务的未提交的更新数据,即读取到了脏数据
不可重复读(unrepeatable read):对同一记录两次读取不一致,因为另一事务对该记录做了修改;
幻读(phantom read):对同一张表的两次查询不一致,因为另一事务插入了一条记录;

不可重复读和幻读的区别!!
不可重复读是读取到了另一事务的更新
幻读是读取到了另一事务的插入(MySQL中无法测试到幻读)

不可重复读:
第一次读:

第二次读:

操作步骤:

CREATE TABLE `person` (
  `pid` int(11) primary key  AUTO_INCREMENT,
  `pname` varchar(20) DEFAULT NULL,
	 pcar varchar(20)
) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8

insert into person(pname,pcar) values('wzj','法拉利');
insert into person(pname,pcar) values('sbt','保时捷');

select * from person; //第一次读
update person set pcar = '兰博基尼' where pid = 5; //另一个事务
select * from person; //第二次读

幻读: 
第一次查询:

第二次查询:

操作步骤:

CREATE TABLE `person` (
  `pid` int(11) primary key  AUTO_INCREMENT,
  `pname` varchar(20) DEFAULT NULL,
	 pcar varchar(20)
) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8

insert into person(pname,pcar) values('wzj','法拉利');
insert into person(pname,pcar) values('sbt','保时捷');

select * from person; //第一次读
insert into person(pname,pcar) values('hg','幻影'); //另一个事务
select * from person; //第二次读


四:事务的隔离级别:
读未提交(read uncommitted)
事务A和事务B,事务A未提交的数据,事务B就可以读取到!,这里读取到的就是脏数据!

读已提交(read committed)
事务A和事务B,只能事务A提交之后的数据,事务B才能读取到
这种隔离级别:可以避免脏读,但会导致:不可重复读

可重复读(repeatable read):Mysql的默认隔离级别
事务A在读到一条数据之后,此时事务B对该数据进行了修改并提交,那么事务A再读该数据,读到的还是原来的内容。
这种隔离级别可以避免脏读和不可重复读,从而达到可重复读的效果,但是会导致幻读!

serializable(串行化)
事务A和事务B,事务A在操作数据库表中的数据的时候,事务B只能排队等待
这种隔离基本不会使用,吞吐量太低,不会导致并发隔离问题,所以并不会
产生并发读问题!!

注:事务的一些命令:

设置全局的隔离级别
set global transaction isolation level read uncommitted

mysql8:查看隔离级别
select @@transaction_isolation  
不是mysql8:查看隔离级别
select @@tx_isolation


开始事务:
start transaction;
回滚事务
rollback;
提交
commit;

 五:用cmd演示一下事务隔离级别的效果 !
演示需要用到的一些命令!

演示:需要用到的一下命令!
mysql -uroot -p123456 (这个后面还不能加分号!)
set global transaction isolation level read uncommitted //设置全局的隔离级别
select @@transaction_isolation //查看隔离级别
use exam
select * from test;
update test set  tname = 'zs' where tid = 2

演示完一遍必须 exit 退出 cls 清一下屏幕, 重登就行!
否则设置完隔离基本没效果

演示:读未提交


演示:读已提交

例子:

01: Read uncommitted 读未提交; 公司发工资了,领导把5000元打到singo的账号上,但是该事务并未提交,而singo正好去查看账户,发现工资已经到账,是5000元整,非常高兴。可是不幸的是,领导发现发给singo的工资金额不对,是2000元,于是迅速回滚了事务,修改金额后,将事务提交,最后singo实际的工资只有2000元,singo空欢喜一场。

02:Read committed 读已提交; singo拿着工资卡去消费,系统读取到卡里确实有2000元,而此时她的老婆也正好在网上转账,把singo工资卡的2000元转到另一账户,并在singo之前提交了事务,当singo扣款时,系统检查到singo的工资卡已经没有钱,扣款失败,singo十分纳闷,明明卡里有钱,为何......

03:Repeatable read 重复读 当singo拿着工资卡去消费时,一旦系统开始读取工资卡信息(即事务开始),singo的老婆就不可能对该记录进行修改,也就是singo的老婆不能在此时转账。

04:重复读可能出现幻读: singo的老婆工作在银行部门,她时常通过银行内部系统查看singo的信用卡消费记录。有一天,她正在查询到singo当月信用卡的总消费金额(select sum(amount) from transaction where month = 本月)为80元,而singo此时正好在外面胡吃海塞后在收银台买单,消费1000元,即新增了一条1000元的消费记录(insert transaction ... ),并提交了事务,随后singo的老婆将singo当月信用卡消费的明细打印到A4纸上,却发现消费总额为1080元,singo的老婆很诧异,以为出现了幻觉,幻读就这样产生了。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值