C#中TransactionScope的使用小結(jié)
TransactionScope是.Net Framework 2.0滯后,新增了一個名稱空間。它的用途是為數(shù)據(jù)庫訪問提供了一個“輕量級”[區(qū)別于:SqlTransaction]的事務(wù)。使用之前必須添加對 System.Transactions.dll 的引用。
下列代碼就是一個正在創(chuàng)建的事務(wù),這個事務(wù)自身還封裝了多個數(shù)據(jù)庫查詢。只要任意一個 SqlCommand 對象引發(fā)異常,程序流控制就會跳出 TransactionScope 的 using 語句塊,隨后,TransactionScope 將自行釋放并回滾該事務(wù)。由于這段代碼使用了 using 語句,所以 SqlConnection 對象和 TransactionScope 對象都將被自動調(diào)用Dispose()釋放。由此可見,只需添加很少的幾行代碼,您就可以構(gòu)建出一個事務(wù)模型,這個模型可以對異常進行處理,執(zhí)行結(jié)束后會 自行清理,此外,它還可以對命令的提交或回滾進行管理。
//創(chuàng)建TransactionScope
using (TransactionScope tsCope= new TransactionScope())
{
using (SqlConnection cn2005= new SqlConnection(someSql2005))
{
SqlCommand cmd= new SqlCommand(sqlUpdate, cn2005);
cn2005.Open();
cmd.ExecuteNonQuery();
}
using (SqlConnection cn2005= new SqlConnection(anotherSql2005))
{
SqlCommand cmd= new SqlCommand(sqlDelete, cn2005);
cn2005.Open();
cmd.ExecuteNonQuery();
}
tsCope.Complete();
}
連接字符串關(guān)鍵字(Enlist)
SqlConnection.ConnectionString 屬性支持關(guān)鍵字 Enlist,該關(guān)鍵字指示 System.Data.SqlClient 是否將檢測事務(wù)上下文并自動在分布式事務(wù)中登記連接。 如果 Enlist=true,連接將自動在打開的線程的當(dāng)前事務(wù)上下文中登記。 如果 Enlist=false,SqlClient 連接不會與分布式事務(wù)進行交互。 Enlist 的默認值為 true。 如果連接字符串中未指定 Enlist,若在連接打開時檢測到一個,連接將自動在分布式事務(wù)中登記。
Server=(local)SQL2005;Database=Northwind;Integrated Security=SSPI;enlist=false
上面所看到的示例中我們使用了TransactionScope的默認設(shè)置。TransactionScope有三種模式:
TransactionScopeOptions
描述
Required
如果已經(jīng)存在一個事務(wù),那么這個事務(wù)范圍將加入已有的事務(wù)。否則,它將創(chuàng)建自己的事務(wù)。
RequiresNew
這個事務(wù)范圍將創(chuàng)建自己的事務(wù)。
Suppress
如果處于當(dāng)前活動事務(wù)范圍內(nèi),那么這個事務(wù)范圍既不會加入氛圍事務(wù) (ambient transaction),也不會創(chuàng)建自己的事務(wù)。當(dāng)部分代碼需要留在事務(wù)外部時,可以使用該選項。
您可以在代碼的任何位置上隨是查看是否存在事務(wù)范圍,具體方法就是查看 System.Transactions.Transaction.Current 屬性。如果這個屬性為“null”,說明不存在當(dāng)前事務(wù)。
若要更改 TransactionScope 類的默認設(shè)置,您可以創(chuàng)建一個 TransactionOptions 對象,然后通過它在 TransactionScope 對象上設(shè)置隔離級別和事務(wù)的超時時間。TransactionOptions 類有一個 IsolationLevel 屬性,通過這個屬性可以更改隔離級別,例如從默認的可序列化 (Serializable) 改為ReadCommitted,甚至可以改為 SQL Server 2005 引入的新的快照 (Snapshot) 級別。(請記住,隔離級別僅僅是一個建議。大多數(shù)數(shù)據(jù)庫引擎會試著使用建議的隔離級別,但也可能選擇其他級別。)此 外,TransactionOptions 類還有一個 TimeOut 屬性,這個屬性可以用來更改超時時間(默認設(shè)置為 1 分鐘)。
下列代碼中使用了默認的 TransactionScope 對象及其默認構(gòu)造函數(shù)。也就是說,它的隔離級別設(shè)置為可序列化 (Serializable),事務(wù)的超時時間為 1 分鐘,而且 TransactionScopeOptions 的設(shè)置為 Required。
TransactionOptions tOpt= new TransactionOptions();
//設(shè)置TransactionOptions模式
tOpt.IsolationLevel= IsolationLevel.ReadCommitted;
// 設(shè)置超時間隔為2分鐘,默認為60秒
tOpt.Timeout= new TimeSpan(0,2,0);
string cnString= ConfigurationManager.ConnectionStrings["sql2005DBServer"].ConnectionString);
using (TransactionScope tsCope= new TransactionScope(TransactionScopeOption.RequiresNew, tOpt))
{
using (SqlConnection cn2005= new SqlConnection(cnString)
{
SqlCommand cmd= new SqlCommand(updateSql1, cn2005);
cn2005.Open();
cmd.ExecuteNonQuery();
}
tsCope.Complete();
}
嵌套應(yīng)用
如下列代碼,假設(shè) Method1 創(chuàng)建一個 TransactionScope,針對一個數(shù)據(jù)庫執(zhí)行一條命令,然后調(diào)用 Method2。Method2 創(chuàng)建一個自身的 TransactionScope,并針對一個數(shù)據(jù)庫執(zhí)行另一條命令。
private void Method1()
{
using (TransactionScope ts=
new TransactionScope(TransactionScopeOption.Required))
{
using (SqlConnection cn2005= new SqlConnection())
{
SqlCommand cmd= new SqlCommand(updateSql1, cn2005);
cn2005.Open();
cmd.ExecuteNonQuery();
}
Method2();
ts.Complete();
}
}
private void Method2()
{
using (TransactionScope ts=
new TransactionScope(TransactionScopeOption.RequiresNew))
{
using (SqlConnection cn2005= new SqlConnection())
{
SqlCommand cmd= new SqlCommand(updateSql2, cn2005);
cn2005.Open();
cmd.ExecuteNonQuery();
}
ts.Complete();
}
}
總結(jié):
進入和退出事務(wù)都要快,這一點非常重要,因為事務(wù)會鎖定寶貴的資源。最佳實踐要求我們在需要使用事務(wù)之前再去創(chuàng)建它,在需要對其執(zhí)行命令前迅速打開連接, 執(zhí)行動作查詢 (Action Query),并盡可能快地完成和釋放事務(wù)。在事務(wù)執(zhí)行期間,您還應(yīng)該避免執(zhí)行任何不必要的、與數(shù)據(jù)庫無關(guān)的代碼,這能夠防止資源被毫無疑義地鎖定過長的 時間。
使用注意:TransactionScope事務(wù)需要開啟一些服務(wù),(如果數(shù)據(jù)庫不在本地,服務(wù)器和客戶端都需要開啟以下服務(wù)和組件服務(wù))
以下組件服務(wù)需要開啟
step1
step2:
相關(guān)文章
聊聊C# 中HashTable與Dictionary的區(qū)別說明
這篇文章主要介紹了聊聊C# 中HashTable與Dictionary的區(qū)別說明,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-01-01c#入門之枚舉和結(jié)構(gòu)體使用詳解(控制臺接收字符串以相反的方向輸出)
這篇文章主要介紹了c#入門之枚舉和結(jié)構(gòu)體使用詳解,最后提供了編寫控制臺應(yīng)用程序接收字符串并做相應(yīng)處理的小示例,需要的朋友可以參考下2014-04-04C# 連接SQL數(shù)據(jù)庫的方法及常用連接字符串
這篇文章主要介紹了C# 連接SQL數(shù)據(jù)庫的方法及常用連接字符串,有需要的朋友可以參考一下2014-01-01