跨数据库(SQL、Oracle)分布式事务
来源:互联网 发布:手机买火车票软件 编辑:程序博客网 时间:2024/06/05 19:03
KB-.NET 2.0 分散式交易新利器---TransactionScope
古老的ASP時代,要做Distributed Transaction(分散式交易,指跨越異質資料庫的交易,例如: 將SQL Server跟Oracle的更新動作包成一個Transaction),有個偷懶的方法。在ASP最前端宣告一下<%@ Transaction=Required%>,則整個ASP中的所有資料庫操作,不管Oracle、SQL、Sybase,通通會自動包成Transaction,不用多寫半行Code。
不過,這種寧可錯殺一百,不可錯放一個的做法效能有點鳥(連沒必要的SELECT動作也被包入Transaction)。會寫VB COM的人多半會寫顆Support Transaction的COM+元件,用來執行特定的資料庫的更新。而多顆異質資料庫的Transactional元件可以再包出一個大Transaction。不過,這得另外寫COM,註冊到COM+ Application中,多了些額外手續。
ASP.NET 1.x誕生後,針對分散式交易,提供了一個四不像的做法:
寫一顆繼承自System.EnterpriseService.ServicedComponent的元件,內含更新資料庫的程式邏輯,再設定TransactionAttribute,然後要Strong-Named/Signed,包上COM+的皮,註冊放入COM+ Application中。
呃... 好像比以前寫COM+還麻煩,堂堂的.NET還是得回頭靠COM+才能實踐分散式交易,會不會有點...
終於.NET 2.0中,針對分散式交易做出了改良。System.Transactions.TransactionScope讓大家有機會重回ASP時代的美妙時光,可以將任意一段資料庫操作包成一個Transaction,不需要額外的手工。例如以下的範例: (要Oracle支援Transaction,記得安裝Oracle Service For Microsoft Transaction Server,不然會百忙一場。)
21 TransactionOptions options =newTransactionOptions();
22 options.IsolationLevel = System.Transactions.IsolationLevel.ReadCommitted;
23 options.Timeout = new TimeSpan(0, 2, 0);
24 using (TransactionScope scope =newTransactionScope(TransactionScopeOption.Required, options))
25 {
26 try
27 {
28 using (SqlConnection sqlCn = new SqlConnection("Data Source=(local); User Id=scott; Password=tigger; Initial Catalog=Lab"))
29 {
30 SqlCommand cmd = new SqlCommand("INSERT INTO tblAccount (Account, Password, Username) VALUES (@acct, @pwd, @name)", sqlCn);
31 cmd.Parameters.Add("@acct",SqlDbType.NVarChar).Value ="EMP" + DateTime.Now.ToString("HHmmss");
32 cmd.Parameters.Add("@pwd",SqlDbType.NVarChar).Value ="PWD";
33 cmd.Parameters.Add("@name",SqlDbType.NVarChar).Value ="NAME" + DateTime.Now.ToString("HHmmss");
34 sqlCn.Open();
35 cmd.ExecuteNonQuery();
36 }
37 //** 實測結果,ODP.NET 9207不Support TransactionScope,必須用.NET 2.0的System.Data.OracleClient
38 using (OracleConnection oraCn = new OracleConnection("Data Source=MYORA; User Id=scott; Password=tigger;"))
39 {
40 OracleCommand cmd = new OracleCommand("INSERT INTO MYTABLE.tblAccount (Account, Password, Username) VALUES (:acct, :pwd, :name)", oraCn);
41 cmd.Parameters.Add(":acct",OracleType.VarChar).Value ="EMP" + DateTime.Now.ToString("HHmmss");
42 cmd.Parameters.Add(":pwd",OracleType.VarChar).Value ="PWD";
43 cmd.Parameters.Add(":name",OracleType.VarChar).Value ="NAME" + DateTime.Now.ToString("HHmmss");
44 oraCn.Open();
45 cmd.ExecuteNonQuery();
46 }
47 //在scope.Complete();後才算Commit!
48 scope.Complete();
49 }
50 catch (Exception ex)
51 {
52 //只要沒有scope.Complete(),先前的動作都會Rollback
53 Response.Write(ex.Message);
54 }
55 }
很簡短吧? (有寫過ServicedComponent的人才能感受出它的好呀!!) 為了證實Transaction效果,我在48行設Breakpoint,則中斷未scope.Complete前,SQL的tblAccount會被鎖定無法SELECT,而ORACLE中則SELECT不到新增的資料(可見SQL用的是Lock大法,ORACLE專攻Snapshot);scope.Complete後,二者的新資料就都出現了。
同時,我還試了故意ORACLE新增動作失敗或不做scope.Complete(),則SQL的tblAccount的新增資料就不會出現。由此,可以驗證以上的程式的確實踐了分散式交易!! 而TransactionScope的確比.NET 1.x的ServicedComponent方便多了,大幅減少異質資料庫包成交易的複雜度。
我曾試著用ODP.NET 9207取代System.Data.OracleClient,測試結果顯示ODP.NET 9207無法參與TrasactionScope物件所建立的交易。後來找到Oracle的ODP.NET FAQ,提到了從ODP.NET 10.2.0.2.20起才支援.NET 2.0的System.Transactions,想用ODP.NET的人要留意。
最後還有一點要注意,如果你的SQL在遠端主機上,用的又是Windows 2003平台,則還有好幾個關節要打通。包含了MSDTC在Windows 2003 SP1上有些選項要調整。還有,你可能會連線失敗,並得到以下這類訊息:
Communication with the underlying transaction manager has failed.
The transaction has already been implicitly or explicitly committed or aborted.
經驗中多半是防火牆的傑作,我的私房解法是在Windows Firewall上開放MSDTC.EXE程式的所有對外連線,問題就可解決。微軟有篇專題文章,介紹MSDTC與Firewall間的愛恨情仇,有興趣的人也可以去挖挖寶。
原帖:http://blog.darkthread.net/blogs/darkthreadtw/archive/2006/11/23/kb-net-2-0-transactionscope.aspx
- 跨数据库(SQL、Oracle)分布式事务
- Oracle 数据库分库分表 分布式事务。
- 跨数据库分布式实时事务
- oracle数据库(事务和动态SQL)
- sql server 连 oracle 'MSDAORA' 无法启动分布式事务
- oracle分布式事务总结
- oracle java 分布式事务
- Oracle分布式事务介绍
- oracle分布式事务总结
- oracle分布式事务总结
- SQL Server分布式事务配置(MSDTC)(远程数据库更新)
- SQL Server分布式事务配置(MSDTC)(远程数据库更新)
- SQL Server分布式事务配置(MSDTC)(远程数据库更新)
- Sql Server分布式事务
- SQL Server 分布式事务
- 分布式数据库跨服务器事务配置(MSDTC)
- 数据库分布式事务感想
- 数据库的分布式事务
- django使用ajax post数据403错误解决方法 (tip)
- gcc编译过程迁徙
- Activity与Thread之间的通讯
- QC与WIN7、IE8 兼容问题解决方案
- oracle sql 语句 in 与EXIST用法比较
- 跨数据库(SQL、Oracle)分布式事务
- JAVA多线程,面试者的最爱!
- iphone学习笔记 – UIView
- MongoDB添加用户-增/删/改/查/权限设置
- Android架构---学东西还是从高处看才能学得远!!
- /Android面试之---讲一下你对Activity的了解(面试必问的).
- SQL的两种分页方式
- 简单内存泄漏检测方法 解决Detected memory leaks!
- 专访实战专家,揭秘iOS神奇开发之路