《SQL必知必会》第二十课 管理事务处理 利用COMMIT和ROLLBACK语句管理事务处理

第二十课 管理事务处理

#事务处理是相当重要的主题
#事务是必须完整执行的SQL语句块
#使用COMMIT和ROLLBACK语句对何时写数据、何时撤销进行明确的管理
#使用保留点,更好地控制回退操作
#各种DBMS对事务处理的实现不同

一、事务处理

#使用事务处理(transaction processing),通过确保成批的SQL操作要么完全执行,要么完全不执行,来维护数据库的完整性

#事务处理是一种机制用来管理必须成批执行的SQL操作,保证数据库不包含不完整的操作结果

#利用事务处理,可以保证一组操作不会中途停止,它们要么完全执行,要么完全不执行(除非明确指示)

#如果没有错误发生,整组语句提交给(写到)数据库表;如果发生错误,则进行回退(撤销),将数据库恢复到某个已知且安全的状态

#例子,这次我们说明过程如何工作。
(1)检查数据库中是否存在相应的客户,如果不存在,添加他。
(2)提交客户信息
(3)检索客户的ID
(4)添加一行到orders表
(5)如果在添加行到orders表时出现故障,回退
(6)检索orders表中赋予的新订单ID
(7)对于订购的每项物品,添加新行到orderitems表
(8)如果在添加新行到Orderitems时出现故障,回退所有添加的orderitems行和orders行

使用事务和事务处理时,有几个反复出现的关键词:

  • 事务(transaction)指一组SQL语句
  • 回退(rollback)指撤销指定SQL语句的过程
  • 提交(commit)指将未存储的SQL语句结果写入数据库表
  • 保留点(savepoint)指事务处理中设置的临时占位符(placeholder),可以对它发布回退(与回退整个事务处理不同)

注意1:
可以回退的语句:
[1]事务处理用来管理INSERT、UPDATE和DELETE语句
[2] 不能回退SELECT语句(回退SELECT语句也没有必要),也不能回退CREATE或DROP操作
[3] 事务处理中可以使用这些语句,但进行回退时,这些操作也不撤销

二、控制事务处理

注意2:
事务处理实现的差异:
#不同DBMS用来实现事务处理的语法有所不同

管理事务的关键在于将SQL语句组分解为逻辑块,并明确规定数据何时应该回退,何时不应该回退
#有的DBMS要求明确标识事务处理块的开始和结束
#如在SQL Server中,标识如下:

【1】BEGIN TRANSACTION ... COMMIT TRANSACTION
#在这个例子中,BEGIN TRANSACTION和COMMIT TRANSACTION语句之间的SQL必须完全执行或者完全不执行

MariaDB、MySQL中等同的代码:
【2】START TRANSACTION;

Oracle中等同的代码:
【3】SET TRANSACTION

PostgreSQL、ANSI SQL中等同的代码:
【4】BEGIN ...
#其他DBMS采用上述语法的变体

多数实现没有明确标识事务处理在何处结束,事务一直存在,直到被中断
通常,COMMIT用于保存更改,ROLLBACK用于撤销

2.1 使用ROLLBACK

#SQL的ROLLBACK命令用来回退(撤销)SQL语句

【5】DELETE FROM Orders; ROLLBACK;
#此例执行DELETE操作,然后用ROLLBACK语句撤销
#这不是最有用的例子,但它的确能够说明,在事务处理块中,DELETE操作(与INSERT和UPDATE操作一样)并不是最终的结果

2.2 使用COMMIT语句

#一般的SQL语句都是针对数据库表直接执行和编写的。这就是所谓的隐式提交(implicit commit),即提交(写或保存)操作是自动进行的

#在事务处理块中,提交不会隐式进行。
#不过,不同DBMS的做法有所不同。有的DBMS按隐式提交处理事务端,有的则不这样

进行明确的提交,使用COMMIT语句:
【6】BEGIN TRANSACTION DELETE OrderItems WHERE order_num = 12345 DELETE Orders WHERE order_num = 12345 COMMIT TRANSACTION
#在这个SQL Server例子中,从系统中完全删除订单12345
#因为涉及更新两个数据库表Orders和OrderItems,所以使用事务处理块来保证订单不被部分删除
#最后的COMMIT语句仅在不出错时写出更改
#如果第一条DELETE起作用,但第二条失败,则DELETE不会提交

【7】SET TRANSACTION DELETE OrderItems WHERE order_num = 12345 DELETE Orders WHERE order_num = 12345 COMMIT;
#Oracle完成上述相同工作

2.3 使用保留点

#使用简单的ROLLBACK和COMMIT语句,就可以写入或撤销整个事务
#但是只对简单的事务才能这样做,复杂的事务可能需要部分提交或回退
#如前面描述的添加订单的过程就是一个事务。如果发生错误,只需要返回到添加orders行之前即可,不需要回退到customers表(如果存在的话)

要支持回退部分事务,必须在事务处理块中的合适位置放置占位符

这样如果需要回退,可以回退到某个占位符

在SQL中,这些占位符称为保留点

在MariaDB、MySQL和Oracle中创建占位符,可使用SAVEPOINT语句:
【8】SAVEPOINT delete1;
#每个保留点都要取能够标识它的唯一名字,以便在回退时,DBMS知道回退到何处

【9】SAVE TRANSACTION delete1;
#在SQL Server中完成上述相同工作

【10】ROLLBACK TRANSACTION delete1;
#回退到本例给出的保留点,在SQL Server中进行

【11】ROLLBACK TO delete1;
#MariaDB、MySQL和Oracle中上述操作如下进行

【12】BEGIN TRANSACTION INSERT INTO Customers(cust_id, cust_name) VALUES('1000000010', 'Toys Emporium'); SAVE TRANSACTION StartOrder; INSERT INTO Orders(order_num, order_date, cust_id) VALUES(20100,'2001/12/1','1000000010'); IF @@ERROR <> 0 ROLLBACK TRANSACTION StartOrder; INSERT INTO OrderItems(order_num, order_item, prod_id, quantity, item_price) VALUES(20100, 1, 'BR01', 100, 5.49); IF @@ERROR <> 0 ROLLBACK TRANSACTION StartOrder; INSERT INTO OrderItems(order_num, order_item, prod_id, quantity, item_price) VALUES(20100, 2, 'BR03', 100, 10.99); IF @@ERROR <> 0 ROLLBACK TRANSACTION StartOrder; COMMIT TRANSACTION
#SQL Server完整例子
#事务处理块中包含了4条INSERT语句
#在第一条INSERT语句之后定义了一个保留点,因此,如果后面的任何一个INSERT操作失败,事务处理能够回退到这里
#在SQL Server中,可检查一个名为@@ERROR的变量,看操作是否成功(其他DBMS使用不同的函数或变量返回此信息)
#如果@@ERROR返回一个非0的值,表示有错误发生,事务处理回退到保留点。如果整个事务处理成功,发布COMMIT以保留数据

注意3:
保留点越多越好:
#可以在SQL代码中设置任意多的保留点,越多越好
#因为保留点越多,就越能灵活地进行回退

BEGIN TRANSACTION 
... 
COMMIT TRANSACTION
--SQL Server

START TRANSACTION; 
--MariaDB、MySQL

SET TRANSACTION
--Oracle

BEGIN 
...
--PostgreSQL、ANSI SQL

DELETE FROM Orders; 
ROLLBACK;

BEGIN TRANSACTION 
DELETE OrderItems WHERE order_num = 12345 
DELETE Orders WHERE order_num = 12345 
COMMIT TRANSACTION
--SQL Server

SET TRANSACTION 
DELETE OrderItems WHERE order_num = 12345 
DELETE Orders WHERE order_num = 12345 
COMMIT;
----Oracle

SAVEPOINT delete1;
--MariaDB、MySQL和Oracle
ROLLBACK TO delete1;
--MariaDB、MySQL和Oracle

SAVE TRANSACTION delete1;
--SQL Server
ROLLBACK TRANSACTION delete1;
--SQL Server

BEGIN TRANSACTION 
INSERT INTO Customers(cust_id, cust_name) 
VALUES('1000000010', 'Toys Emporium'); 
SAVE TRANSACTION StartOrder; 
INSERT INTO Orders(order_num, order_date, cust_id) 
VALUES(20100,'2001/12/1','1000000010'); 
IF @@ERROR <> 0 ROLLBACK TRANSACTION StartOrder; 
INSERT INTO OrderItems(order_num, order_item, prod_id, quantity, item_price) 
VALUES(20100, 1, 'BR01', 100, 5.49); 
IF @@ERROR <> 0 ROLLBACK TRANSACTION StartOrder; 
INSERT INTO OrderItems(order_num, order_item, prod_id, quantity, item_price) 
VALUES(20100, 2, 'BR03', 100, 10.99); 
IF @@ERROR <> 0 ROLLBACK TRANSACTION StartOrder; 
COMMIT TRANSACTION
--SQL Server




【1】《SQL必知必会》第四版 人民邮电出版社 [美] Ben Forta 著 钟鸣 刘晓霞 译
【2】https://www.cnblogs.com/sanjun/p/8294958.html
【3】https://blog.csdn.net/qq_41007611/article/details/107356113

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值