TCL事务控制语言

1. TCL事务控制语言

1.1 事务的创建及特征

事务:一个或一组sql语句组成一个执行单元,这个执行单元要么全部执行,要么全部不执行。

事务的特性:

  • 原子性:一个事务不可再分割,要么都执行要么都不执行。
  • 一致性:一个事务执行会使数据从一个一致状态切换到另外一个一致状态。
  • 隔离性:一个事务的执行不受其他事务的干扰。
  • 持久性:一个事务一旦提交,则会永久的改变数据库的数据。

事务的创建:

  • 隐式事务:事务没有明显的开启和结束的标记,比如insert、update、delete语句。
  • 显式事务:事务具有明显的开启和结束的标记(必须先设置自动提交功能为禁用)。
#1.演示事务的使用步骤
SHOW VARIABLES LIKE 'autocommit';
SHOW ENGINES;

#开启事务
SET autocommit=0;
START TRANSACTION;
#编写一组事务的语句
UPDATE account SET balance = 1000 WHERE username='张三';
UPDATE account SET balance = 1000 WHERE username='李四';
#结束事务
ROLLBACK;
#COMMIT;

1.2 事务的隔离级别

事务的隔离级别:

  • read uncommitted(读未提交):

    如果一个事务已经开始写数据,则另外一个事务不允许同时对该数据进行写操作,但允许其他事务读此行数据。该隔离级别可以通过“排他写锁”,但是不排斥读线程实现。这样就避免了更新丢失,却可能出现脏读,也就是说事务B读取到了事务A未提交的数据。解决了更新丢失,但还是可能会出现脏读。

  • read committed(读提交):

    如果是一个读事务正在读数据,则允许其他事务对该数据进行读写,如果是写事务将会禁止其他事务访问该行数据。该隔离级别避免了脏读,但是可能出现不可重复读,也就是说事务A事先读取了数据,事务B紧接着更新了数据,并提交了事务,而事务A再次读取该数据时,数据已经发生了改变。解决了更新丢失和脏读问题。

  • repeatable read(可重复读取):
    可重复读取是指在一个事务内,多次读同一个数据,在这个事务还没结束时,其他事务不能访问该数据(包括了读写),这样就可以在同一个事务内两次读到的数据是一样的,因此称为是可重复读隔离级别。读取数据的事务将会禁止对该数据进行写事务(但允许读事务),写事务则禁止任何其他事务(包括了读写),这样避免了不可重复读和脏读,但是有时可能会出现幻读。(读取数据的事务)可以通过“共享读镜”和“排他写锁”实现。解决了更新丢失、脏读、不可重复读、但是还会出现幻读。

    幻读和不可重复读都是在同一个事务中多次读取了其他事务已经提交的事务的数据导致每次读取的数据不一致,但不可重复读重点在于update和delete,而幻读的重点在于insert。如果使用锁机制来实现这两种隔离级别,在可重复读中,该sql第一次读取到数据后,就将这些数据加锁,其它事务无法修改这些数据,就可以实现可重复读了。但这种方法却无法锁住insert的数据,所以当事务A先前读取了数据,事务B还是可以insert数据提交,然后事务A修改了全部数据,这时事务A就会发现莫名其妙多了一些之前没有的数据,这就是幻读。举个例子,事务A和事务B同时开始,事务A插入了一条数据(假如原本有4条数据),并且提交了事务,在事务A提交事务之后,事务B中查看只有4条数据,此时执行一条更新全表的语句,提示却发现更新了5条数据。

  • Serializable(可序化):
    提供严格的事务隔离,它要求事务序列化执行,事务只能一个接着一个地执行,仅仅通过“行级锁”是无法实现序列化的,必须通过其他机制保证新插入的数据不会被执行查询操作的事务访问到。序列化是最高的事务隔离级别,同时代价也是最高的,性能很低,一般很少使用,在该级别下,事务顺序执行,不仅可以避免脏读、不可重复读,还避免了幻读。解决了更新丢失、脏读、不可重复读、幻读(虚读)。

脏读不可重复读幻读
read uncommitted
read committed
repeatable read
serializable

mysql中默认第三个隔离级别:repeatable read,oracle中默认第二个隔离级别:read committed。

查看隔离级别:select @@tx_isolation;

设置隔离级别:set session|global transaction isolation level 隔离级别;

1.3 savepoint

#演示savepoint的使用
SET autocommit=0;
START TRANSACTION;
DELETE FROM account WHERE id=25;
SAVEPOINT a; -- 设置保存点
DELETE FROM account WHERE id=28;
ROLLBACK TO a; -- 回滚到保存点(id=28的记录没有被删除,id=25的记录被删除了)
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值