oracle使用語句savepoint sp_begintran開啟顯式事務(wù),鎖本身和事務(wù)是沒有關(guān)系的,只要是數(shù)據(jù)庫的操作都會(huì)產(chǎn)生鎖。處于事務(wù)中的SQL語句只有這個(gè)事務(wù)提交(commit)之后,事務(wù)中的SQL語句影響的表記錄上的鎖才會(huì)釋放。鎖常見有共享鎖(select語句產(chǎn)生)和排它鎖(DML語句產(chǎn)生),如果一個(gè)表上加載有共享鎖,還可以疊加共享鎖,但不能疊加排它鎖。如果一個(gè)表上加載有排他鎖,就什么鎖都不能加了,也就是說如果DML語句占用過多的時(shí)間,這些數(shù)據(jù)庫效率就不高,就需要優(yōu)化,當(dāng)然select語句性能低了也不行。
創(chuàng)新互聯(lián)建站-專業(yè)網(wǎng)站定制、快速模板網(wǎng)站建設(shè)、高性價(jià)比南潯網(wǎng)站開發(fā)、企業(yè)建站全套包干低至880元,成熟完善的模板庫,直接使用。一站式南潯網(wǎng)站制作公司更省心,省錢,快速模板網(wǎng)站建設(shè)找我們,業(yè)務(wù)覆蓋南潯地區(qū)。費(fèi)用合理售后完善,10余年實(shí)體公司更值得信賴。
每個(gè)存儲(chǔ)過程可以不用顯式事務(wù),它本身就為你開啟了一個(gè)隱式事務(wù),如果需要開啟顯示事務(wù),就通過savepoint sp_begintran開啟,無論是不是顯式還是隱式事務(wù),你都得通過commit work提交事務(wù),通過exception捕捉SQL語句異常,在異常發(fā)生時(shí)需要回滾事務(wù)(rollback work)。
事務(wù)是為了實(shí)現(xiàn)數(shù)據(jù)的一致性和并發(fā)處理,你對(duì)數(shù)據(jù)進(jìn)行定義和修改時(shí)就會(huì)產(chǎn)生事務(wù)。
例如你執(zhí)行update語句,這時(shí)事務(wù)就開始了,執(zhí)行commit會(huì)提交事務(wù),而rollback會(huì)回滾(撤銷)事務(wù)。如果你什么也不做,那么正常退出程序時(shí)會(huì)自動(dòng)提交,而程序異常中止則會(huì)回滾事務(wù)。
事務(wù)是為了實(shí)現(xiàn)數(shù)據(jù)的一致性和并發(fā)處理,你對(duì)數(shù)據(jù)進(jìn)行定義和修改時(shí)就會(huì)產(chǎn)生事務(wù)。
例如你執(zhí)行update語句,這時(shí)事務(wù)就開始了,執(zhí)行commit會(huì)提交事務(wù),而rollback會(huì)回滾(撤銷)事務(wù)。如果你什么也不做,那么正常退出程序時(shí)會(huì)自動(dòng)提交,而程序異常中止則會(huì)回滾事務(wù)。
BeginTrans用于開始一個(gè)事物;RollbackTrans用于回滾事務(wù);CommitTrans用于提交所有的事務(wù)處理結(jié)果,即確認(rèn)事務(wù)的處理
BeginTrans和CommitTrans 用于標(biāo)記事務(wù)的開始和結(jié)束,在這兩個(gè)之間的語句,就是作為事務(wù)處理的語句
事務(wù)隔離級(jí)別:一個(gè)事務(wù)對(duì)數(shù)據(jù)庫的修改與并行的另一個(gè)事務(wù)的隔離程度。
兩個(gè)并發(fā)事務(wù)同時(shí)訪問數(shù)據(jù)庫表相同的行時(shí),可能存在以下三個(gè)問題:
1、幻想讀:事務(wù)T1讀取一條指定where條件的語句,返回結(jié)果集。此時(shí)事務(wù)T2插入一行新記錄,恰好滿足T1的where條件。然后T1使用相同的條件再次查詢,結(jié)果集中可以看到T2插入的記錄,這條新紀(jì)錄就是幻想。
2、不可重復(fù)讀?。菏聞?wù)T1讀取一行記錄,緊接著事務(wù)T2修改了T1剛剛讀取的記錄,然后T1再次查詢,發(fā)現(xiàn)與第一次讀取的記錄不同,這稱為不可重復(fù)讀。
3、臟讀:事務(wù)T1更新了一行記錄,還未提交所做的修改,這個(gè)T2讀取了更新后的數(shù)據(jù),然后T1執(zhí)行回滾操作,取消剛才的修改,所以T2所讀取的行就無效,也就是臟數(shù)據(jù)。
為了處理這些問題,SQL標(biāo)準(zhǔn)定義了以下幾種事務(wù)隔離級(jí)別
READ?UNCOMMITTED?幻想讀、不可重復(fù)讀和臟讀都允許。
READ?COMMITTED?允許幻想讀、不可重復(fù)讀,不允許臟讀
REPEATABLE?READ?允許幻想讀,不允許不可重復(fù)讀和臟讀
SERIALIZABLE?幻想讀、不可重復(fù)讀和臟讀都不允許
Oracle數(shù)據(jù)庫支持READ?COMMITTED?和?SERIALIZABLE這兩種事務(wù)隔離級(jí)別。所以O(shè)racle不支持臟讀
SQL標(biāo)準(zhǔn)所定義的默認(rèn)事務(wù)隔離級(jí)別是SERIALIZABLE,但是Oracle?默認(rèn)使用的是READ?COMMITTED
設(shè)置隔離級(jí)別使用?SET?TRANSACTION?ISOLATION?LEVEL?[READ?UNCOMMITTED|READ?COMMITTED|REPEATABLE?READ|SERIALIZABLE]
--?下面是oracle?設(shè)置SERIALIZABLE隔離級(jí)別一個(gè)示例:
左面是事務(wù)T1,右面是事務(wù)T2,因?yàn)門2級(jí)別為SERIALIZABLE,所以即使事務(wù)T1在提交了數(shù)據(jù)之后,事務(wù)T2還是看不到T1提交的數(shù)據(jù),幻想讀和不可重復(fù)讀都不允許了。
那如何能查看到T1新增的記錄呢? 上面T1和T2是并發(fā)執(zhí)行,在T1執(zhí)行insert的時(shí)候事務(wù)T2已經(jīng)開始了,因?yàn)門2級(jí)別是SERIALIZABLE,所以T2所查詢的數(shù)據(jù)集是T2事務(wù)開始前數(shù)據(jù)庫的數(shù)據(jù)。即事務(wù)T1在事務(wù)T2開始之后的insert和update操作的影響都不會(huì)影響事務(wù)T2?,F(xiàn)在重新開啟一個(gè)事務(wù)T3 就可以看到T1新增的記錄了。
當(dāng)下列事件發(fā)生時(shí),事務(wù)就開始了:
1、連接到數(shù)據(jù)庫,并執(zhí)行第一條DML語句
2、前一個(gè)事務(wù)結(jié)束后,又輸入了另一條DML語句
在C# 和 VB.NET 中使用數(shù)據(jù)庫事務(wù)您可以使用 OracleTransaction 類的一個(gè)對(duì)象來表示一個(gè)事務(wù)。OracleTransaction 類包含多個(gè)屬性,其中的兩個(gè)為 Connection(指定與事務(wù)關(guān)聯(lián)的數(shù)據(jù)庫連接)和 IsolationLevel(指定事務(wù)隔離級(jí)別);本文稍后將向您介紹更多有關(guān)事務(wù)隔離級(jí)別的內(nèi)容。 OracleTransaction 類包含許多操控事務(wù)的方法。您可以使用 Commit() 方法永久提交 SQL 語句,并可以使用 Rollback() 撤銷這些語句。您還可以使用 Save() 在事務(wù)中設(shè)置一個(gè)保存點(diǎn)。 我現(xiàn)在將帶著您逐步完成兩個(gè)示例程序 ― 一個(gè)用 C# 編寫 (TransExample1.cs),另一個(gè)用 VB.NET 編寫 (TransExample1.vb)。這些程序演示了如何執(zhí)行一個(gè)包含了兩條 INSERT 語句的事務(wù)。第一條 INSERT 語句將在表 product_types 中添加一行,第二條將在表 products 中添加一行。 導(dǎo)入命名空間以下C# 程序語句指定在程序中使用 System 和 Oracle.DataAcess.Client 命名空間: using System; using Oracle.DataAccess.Client;下面是等價(jià)的 VB.NET 語句: Imports System Imports Oracle.DataAccess.ClientOracle.DataAccess.Client 命名空間是 ODP.NET 的一部分,它包含許多類,其中有OracleConnection、OracleCommand 和 OracleTransaction。示例程序用到了這些類。 第1 步 創(chuàng)建一個(gè) OracleConnection 對(duì)象連接到 Oracle 數(shù)據(jù)庫,然后打開該連接。 在C# 中: OracleConnection myOracleConnection = new OracleConnection( "User Id=store;Password=store;Data Source=ORCL" ); myOracleConnection.Open();在VB.NET 中: Dim myOracleConnection As New OracleConnection( _ "User Id=store;Password=store;Data Source=ORCL") myOracleConnection.Open()User Id 和 Password 屬性指定了您所要連接到的模式的數(shù)據(jù)庫用戶和口令。Data Source 屬性指定了數(shù)據(jù)庫的 Oracle Net 服務(wù)名稱;初始數(shù)據(jù)庫的默認(rèn)服務(wù)名稱為 ORCL。如果您使用的不是初始數(shù)據(jù)庫,或者您的服務(wù)名稱不同,那么您需要在程序中修改 Data Source 屬性的設(shè)置。 第2 步 創(chuàng)建一個(gè) OracleTransaction 對(duì)象,然后調(diào)用 OracleConnection 對(duì)象的 BeginTransaction() 方法啟動(dòng)事務(wù)。 在C# 中: OracleTransaction myOracleTransaction = myOracleConnection.BeginTransaction(); In VB.NET: Dim myOracleTransaction As OracleTransaction = _ myOracleConnection.BeginTransaction()第3 步 創(chuàng)建一個(gè) OracleCommand 對(duì)象,用于存儲(chǔ) SQL 語句。 在C# 中: OracleCommand myOracleCommand = myOracleConnection.CreateCommand();在VB.NET 中: Dim myOracleCommand As OracleCommand = myOracleConnection.CreateCommand因?yàn)镺racleCommand 對(duì)象使用 OracleConnection 對(duì)象的 CreateCommand() 方法創(chuàng)建的,所以它自動(dòng)使用在第 2 步中為 OracleConnection 對(duì)象設(shè)置的事務(wù)。 第4 步 將OracleCommand 對(duì)象的 CommandText 屬性設(shè)為向表 product_types 中添加一行的第一條 INSERT 語句。 在C# 中: myOracleCommand.CommandText = "INSERT INTO product_types (" + " product_type_id, name" + ") VALUES (" + " 3, 'Magazine'" + ")";在VB.NET 中: myOracleCommand.CommandText = _ "INSERT INTO product_types (" _ " product_type_id, name" _ ") VALUES (" _ " 3, 'Magazine'" _ ")"