事务

举报
xcc-2022 发表于 2022/11/28 19:07:21 2022/11/28
【摘要】 16、事务 16.1  、概述事务可以保证多个操作原子性,要么全成功,要么全失败。对于数据库来说事务保证批量的 DML 要么全成功,要么全失败。事务具有四个特征ACIDa)         原子性(Atomicity)l  整个事务中的所有操作,必须作为一个单元全部完成(或全部取消)。b)         一致性(Consistency)l  在事务开始之前与结束之后,数据库都保持一致状态。...

16、事务

 

16.1 


、概述

事务可以保证多个操作原子性,要么全成功,要么全失败。对于数据库来说事务保证批量的 DML 要么全成功,要么全失败。事务具有四个特征ACID

a)         原子性(Atomicity)

l  整个事务中的所有操作,必须作为一个单元全部完成(或全部取消)。

b)         一致性(Consistency)

l  在事务开始之前与结束之后,数据库都保持一致状态。

c)         隔离性(Isolation)

l  一个事务不会影响其他事务的运行。

d)         持久性(Durability)

l  在事务完成以后,该事务对数据库所作的更改将持久地保存在数据库之中,并不会被回滚。事务中存在一些概念:

a)         事务(Transaction):一批操作(一组 DML)

b)         开启事务(Start Transaction)

c)         回滚事务(rollback)

d)         提交事务(commit)

e)         SET AUTOCOMMIT:禁用或启用事务的自动提交模式

 

当执行 DML 语句是其实就是开启一个事务

关于事务的回滚需要注意:只能回滚 insert、delete 和 update 语句,不能回滚 select(回滚select 没有任何意义),对于

createdropalter 这些无法回滚.


事务只对 DML 有效果。

 

注意:rollback,或者commit 后事务就结束了。

 

16.2  、事务的提交与回滚演示

 

1)             创建表

 

create table user(

id int (11) primary key not null auto_increment , username varchar(30),

password varchar(30)

)    ENGINE=InnoDB DEFAULT CHARSET=utf8

 

 

2)             查询表中数据


 

3)             开启事务 START TRANSACTION;

4)             插入数据

insert into user (username,password) values ('zhangsan','123');


 

5)             查看数据


6)             修改数据


 

7)             查看数据


 


 

8)             回滚事务


 

 

9)             查看数据


 

16.3  、自动提交模式

 

•       自动提交模式用于决定新事务如何及何时启动。

•       启用自动提交模式:

–       如果自动提交模式被启用,则单条DML 语句将缺省地开始一个新的事务。

–       如果该语句执行成功,事务将自动提交,并永久地保存该语句的执行结果。

–       如果语句执行失败,事务将自动回滚,并取消该语句的结果。

–       在自动提交模式下,仍可使用 START TRANSACTION 语句来显式地启动事务。这时,一个事务仍可包含多条语句,直到这些语句被统一提交或回滚。

•       禁用自动提交模式:

–       如果禁用自动提交,事务可以跨越多条语句。

–       在这种情况下,事务可以用 COMMIT 和 ROLLBACK 语句来显式地提交或回滚。

•       自动提交模式可以通过服务器变量AUTOCOMMIT 来控制。

•       例如:

mysql> SET AUTOCOMMIT = OFFmysql> SET AUTOCOMMIT = ON

mysql> SET SESSION AUTOCOMMIT = OFFmysql> SET SESSION AUTOCOMMIT = ON

show variables like '%auto%'; -- 查看变量状态


 

16.4  、事务的隔离级别

 

16.4.1    、隔离级别

 

•       事务的隔离级别决定了事务之间可见的级别。

•       当多个客户端并发地访问同一个表时,可能出现下面的一致性问题:

–       脏读取(Dirty Read)

一个事务开始读取了某行数据,但是另外一个事务已经更新了此数据但没有能够及时提交,这就出现了脏读取。

–       不可重复读(Non-repeatable Read)

在同一个事务中,同一个读操作对同一个数据的前后两次读取产生了不同的结果,这就是不可重复读。

–       幻像读(Phantom Read)

幻像读是指在同一个事务中以前没有的行,由于其他事务的提交而出现的新行。

 

16.4.2    、四个隔离级别

 

•       InnoDB 实现了四个隔离级别,用以控制事务所做的修改,并将修改通告至其它并发的事务:

–       读未提交(READ UMCOMMITTED) 允许一个事务可以看到其他事务未提交的修改。

–       读已提交(READ COMMITTED)

允许一个事务只能看到其他事务已经提交的修改,未提交的修改是不可见的。

–       可重复读(REPEATABLE READ)

确保如果在一个事务中执行两次相同的SELECT 语句,都能得到相同的结果,不管其他事务是否提交这些修改。(银行总账)

该隔离级别为 InnoDB 的缺省设置。

–       串行化(SERIALIZABLE) 【序列化】将一个事务与其他事务完全地隔离。

 

例:A 可以开启事物,B 也可以开启事物A 在事物中执行DML 语句时,未提交B 不以执行 DML,DQL 语句


16.4.3    、隔离级别与一致性问题的关系

 


 

 

 

 

16.4.4    、设置服务器缺省隔离级别

 

通过修改配置文件设置

 

•       可以在 my.ini 文件中使用 transaction-isolation 选项来设置服务器的缺省事务隔离级别。

•       该选项值可以是:

–       READ-UNCOMMITTED

–       READ-COMMITTED

–       REPEATABLE-READ

–       SERIALIZABLE

•       例如:

[mysqld]

transaction-isolation = READ-COMMITTED

 

通过命令动态设置隔离级别

 

•       隔离级别也可以在运行的服务器中动态设置,应使用SET TRANSACTION ISOLATION LEVEL 语句。

•       其语法模式为:


SET [GLOBAL | SESSION] TRANSACTION ISOLATIONLEVEL <isolation-level>

其中的<isolation-level>可以是:

–       READ UNCOMMITTED

–       READ COMMITTED

–       REPEATABLE READ

–       SERIALIZABLE

•       例如: SET TRANSACTION ISOLATION LEVEL REPEATABLE READ;

 

16.4.5    、隔离级别的作用范围

 

•       事务隔离级别的作用范围分为两种:

–       全局级:对所有的会话有效

–       会话级:只对当前的会话有效

•       例如,设置会话级隔离级别为READ COMMITTED : mysql> SET TRANSACTION ISOLATION LEVEL READ COMMITTED; 或:

mysql> SET SESSION TRANSACTION ISOLATIONLEVEL READ COMMITTED

•       设置全局级隔离级别为READ COMMITTED :

mysql> SET GLOBAL TRANSACTION ISOLATIONLEVEL READ COMMITTED

 

16.4.6    、查看隔离级别

 

•       服务器变量 tx_isolation(包括会话级和全局级两个变量)中保存着当前的会话隔离级别。

•       为了查看当前隔离级别,可访问tx_isolation 变量:

–       查看会话级的当前隔离级别:

mysql> SELECT @@tx_isolation;

或:

mysql> SELECT @@session.tx_isolation;

–       查看全局级的当前隔离级别:

mysql> SELECT @@global.tx_isolation;

 

16.4.7    、并发事务与隔离级别示例

 

read uncommitted(未提交读) --脏读(Drity Read)

 

 

会话一

会话二

mysql> prompt s1>

mysql> use bjpowernode

s1>use bjpowernode

mysql> prompt s2>


 

s1>create table tx (

id int(11), num int (10)

);

 

s1>set global transaction isolation level read

uncommitted;

 

s1>start transaction;

 

 

s2>start transaction;

s1>insert into tx values (1,10);

 

 

s2>select * from tx;

s1>rollback;

 

 

s2>select * from tx;

 

 

read committed(已提交读)

 

 

会话一

会话二

s1> set global transaction isolation level read

committed;

 

s1>start transaction;

 

 

s2>start transaction;

s1>insert into tx values (1,10);

 

s1>select * from tx;

 

 

s2>select * from tx;

s1>commit;

 

 

s2>select * from tx;

 

repeatable read(可重复读)

 

 

会话一

会话二

s1>  set  global  transaction  isolation  level

repeatable read;

 

s1>start transaction;

s2>start transaction;

s1>select * from tx;

 

s1>insert into tx values (1,10);

 

 

s2>select * from tx;

s1>commit;

 

 

s2>select * from tx;

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。