SQL 管理事务处理

管理事务处理

一、事务处理

使用事务处理(transaction processing),通过确认成批的SQL操作要么完全执行,要么完全不执行,来维护数据库的完整性。
关系数据库把数据存储在多个表中,使数据更容易操作、维护和重用。不用深究如何以及为什么进行关系数据库设计,在某种程序上说,设计良好的数据模式都是关联的。
订单存储在Orders和OrderItems两个表中:Orders存储实际的订单,OrderItems存储订购的各项物品。这两个表使用称为主键的唯一ID互相关联,又与包含客户和产品信息的其他表相关联。
给系统添加订单的过程如下:
(1)检查数据库中是否存在相应的顾客,如果不存在,添加他;
(2)检索顾客的ID;
(3)在Orders表添加一行,它与顾客ID相关联;
(4)检索Orders表中赋予的新订单ID;
(5)为订购的每个物品在OrderItems中添加一行,通过检索出来的ID把它与Orders表相关联(并且通过产品ID与Products表相关联)。

现在假设由于某种数据库故障(如超出磁盘空间、安全限制、表锁等),这个过程无法完成。数据库中的数据会出现什么情况?

如果故障发生在添加顾客之后,添加Orders表之前,则不会有什么问题。某些顾客没有订单是完全合法的。重新执行次过程前时,所插入的顾客记录将被检索和使用。可以有效地从出故障的地方开始执行此过程。

但是,如果故障发生在插入Orders行之后,添加OrderItems行之前,怎么办?现在,数据库中有一个空订单。
更糟的是,如果系统在添加OrderItems 行之时出现故障,怎么办?结果是数据库中存在不完整的订单,而你还不知道。

如何解决这种问题?这就需要使用事务处理了。事务处理是一种机制,用来管理必须成批执行的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),可以对它发布回退(与回退整个事务处理不同)。

提示:可以回退哪些语句?
事务处理用来管理INSERT、UPDATE 和 DELETE 语句。不能回退 SELECT语句(回退 SELECT 语句也没有必要),也不能回退 CREATE 或 DROP 操作。事务处理中可以使用这些语句,但进行回退时,这些操作也不撤销。

二、控制事务处理

我们已经知道了什么是事务处理,下面讨论管理事务中涉及的问题。

注意:事务处理实现的差异
不同DBMS用来实现事务处理的语法有所不同。在使用事务处理时请参阅相应的DBMS文档。

管理事务的关键在于将 SQL 语句组分解为逻辑块,并明确规定数据何时应该回退,何时不应该回

有的 DBMS要求明确标识事务处理块的开始和结束。如在 SQL Server中,标识如下

BEGIN TRANSACTION 
... 
COMMIT TRANSACTION

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

MariaDB 和 MySQL 中等同的代码为

START TRANSACTION 
...

Oracle使用的语法:

SET TRANSACTION 
... 

PostgreSQL 使用 ANSI SQL 语法:

BEGIN 
... 

其他 DBMS 采用上述语法的变体。你会发现,多数实现没有明确标识事务处理在何处结束。事务一直存在,直到被中断。通常,COMMITT 用于保存更改,ROLLBACK 用于撤销,详述如下。

1、使用ROLLBACK

SQL 的 ROLLBACK 命令用来回退(撤销)SQL 语句,请看下面的语句:

DELETE FROM Orders; 
ROLLBACK; 

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

2、使用COMMIT

一般的 SQL 语句都是针对数据库表直接执行和编写的。这就是所谓的隐式提交(implicit commit),即提交(写或保存)操作是自动进行的。
在事务处理块中,提交不会隐式进行。不过,不同 DBMS 的做法有所不同。有的 DBMS 按隐式提交处理事务端,有的则不这样。
进行明确的提交,使用 COMMIT 语句。下面是一个 SQL Server 的例子:

BEGIN TRANSACTION 
DELETE OrderItems WHERE order_num = 12345 
DELETE Orders WHERE order_num = 12345 
COMMIT TRANSACTION 

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

为在Oracle中完成相同的工作,可如下进行:

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

3、使用保留点

使用简单的ROLLBACK 和 COMMIT 语句,就可以写入或撤销整个事务。但是,只对简单的事务才能这样做,复杂的事务可能需要部分提交或回退。

例如前面描述的添加订单的过程就是一个事务。如果发生错误,只需要返回到添加Orders行之前即可。不需要回退到Customers表(如果存在的话)。

要支持回退部分事务,必须在事务处理块中的合适位置放置占位符。这样。如果需要回退,可以回退到某个占位符。

在 SQL 中,这些占位符称为保留点。在 MariaDB、MySQL 和 Oracle 中创建占位符,可使用 SAVEPOINT 语句。

SAVEPOINT delete1; 

在 SQL Server 中,如下进行:

ROLLBACK TO 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

这里的事务处理块中包含了 4 条 INSERT 语句。在第一条 INSERT 语句之后定义了一个保留点,因此,如果后面的任何一个 INSERT 操作失败,事务处理能够回退到这里。在 SQL Server 中,可检查一个名为@@ERROR的变量,看操作是否成功。(其他 DBMS 使用不同的函数或变量返回此信息。)如果@@ERROR 返回一个非 0 的值,表示有错误发生,事务处理回退到保留点。如果整个事务处理成功,发布 COMMIT 以保留数据。

提示:保留点越多越好
可以在 SQL 代码中设置任意多的保留点,越多越好。为什么呢?因为
保留点越多,你就越能灵活地进行回退

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

伟兴竟在思考

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值