11.4.1 事務的由來
使用DELETE 命令或UPDATE 命令對數據庫進行更新時一次只能操作一個表,這會帶來數據庫的數據不一致的問題。例如:企業取消了後勤部,需要將‘後勤部’從department表中刪除,要修改department 表,而employee 表中的部門編號與後勤部相對應的員工也應刪除。因此,兩個表都需要修,改這種修改只能通過兩條DELETE 語句進行。假設後勤部編號為‘1012’
第一條DELETE 語句修改department 表
delete from department
where dept_id = ’1012’
第二條DELETE 語句修改employee 表
delete from employee
where dept_id = ’1012’
在執行第一條DELETE 語句後,數據庫中的數據已處於不一致的狀態,因為此時已經沒有‘後勤部’了,但employee 表中仍然保存著屬於後勤部的員工記錄。只有執行了第二條DELETE 語句後數據才重新處於一致狀態。但是,如果執行完第一條語句後,計算機突然出現故障,無法再繼續執行第二條DELETE 語句,則數據庫中的數據將處於永遠不一致的狀態。因此,必須保證這兩條DELETE 語句同時執行。為解決類似的問題,數據庫系統通常都引入了事務(Transaction) 的概念。
11.4.2 事務的概念
事務是一種機制,是一個操作序列,它包含了一組數據庫操作命令,所有的命令作為一個整體一起向系統提交或撤消操作請求,即要麼都執行,要麼都不執行。因此,事務是一個不可分割的工作邏輯單元,類似於操作系統中的原語。在數據庫系統上執行並發操作時,事務是作為最小的控制單元來使用的。
通常在程序中用BEGIN TRANSACTION 命令來標識一個事務的開始,用COMMITTRANSACTION 命令標識事務結束。這兩個命令之間的所有語句被視為一體,只有執行到COMMIT TRANSACTION 命令時,事務中對數據庫的更新操作才算確認。和BEGIN…END 命令類似,這兩個命令也可以進行嵌套,即事務可以嵌套執行。這兩個命令的語法如下:
BEGIN TRAN[SACTION] [transaction_name | @tran_name_variable]
COMMIT [ TRAN[SACTION] [transaction_name | @tran_name_variable] ]
其中BEGIN TRANSACTION 可以縮寫為BEGIN TRAN、 COMMIT TRANSACTION可以縮寫為COMMIT TRAN 或COMMIT。
transaction_name指定事務的名稱。只有前32 個字符會被系統識別。 @tran_name_variable
用變量來指定事務的名稱,變量只能聲明為CHAR、 VARCHAR、 NCHAR 或 NVARCHAR 類型。
11.4.3 事務回滾
事務回滾(Transaction Rollback) 是指當事務中的某一語句執行失敗時,將對數據庫的操作恢復到事務執行前或某個指定位置。
事務回滾使用ROLLBACK TRANSACTION 命令,其語法如下:
ROLLBACK [TRAN[SACTION] [transaction_name | @tran_name_variable
| savepoint_name | @savepoint_variable] ]
其中savepoint_name 和@savepoint_variable 參數用於指定回滾到某一指定位置。
如果要讓事務回滾到指定位置,則需要在事務中設定保存點(Save Point)。所謂保存點是指定其所在位置之前的事務語句,不能回滾的語句即此語句前面的操作被視為有效。
其語法如下:
SAVE TRAN[SACTION] {savepoint_name | @savepoint_variable}
各參數說明如下:
avepoint_name
指定保存點的名稱。同事務的名稱一樣,只有前32 個字符會被系統識別。
@savepoint_variable
用變量來指定保存點的名稱。變量只能聲明為CHAR、 VARCHAR、 NCHAR 或NVARCHAR 類型。
注意:如果不指定回滾的事務名稱或保存點,則ROLLBACK TRANSACTION命令會將事務回滾到事務執行前,如果事務是嵌套的、則會回滾到最靠近的BEGIN TRANSACTION命令前。