有没有办法创建ADO.NET连接并忽略环境事务?

时间:2009-07-31 21:10:10

标签: c# .net ado.net transactions transactionscope

我遇到的情况是我在一个具有TransactionScopeRequired = true的WCF服务中运行,这意味着总会有一个环境事务。

但是,我需要在应用程序的生命周期内启动一个新的连接,这意味着我不能让它使用abmbient事务。

关于如何做到这一点的任何想法?这样做会自动使用环境事务:

Assert.IsNotNull(System.Transactions.Transaction.Current);
var conn = new OracleConnection("my connection string");
conn.Open(); // <-- picks up ambient transaction, but I don't want that

实际上这个例子可以通过这样说来简化:

OracleConnection conn; // <-- this is actually held around in another object that has a very long lifetime, well past the TransactionScope.
using(var tx = new TransactionScope())
{
    conn = new OracleConnection("my connection string");
    conn.Open(); // <-- picks up ambient transaction, but I don't want that
    // ... do stuff
}

我不希望我的连接真正拿起TransactionScope。在实际的代码中,还有很多内容在范围内执行数据库操作,我只需要在事务范围的生命周期内保留1个。


我猜真实的情况值得一提。这里实际发生的是在WCF服务调用期间,我使用企业库缓存块将对象添加到缓存。此对象是一个数据表,但也保持与Oracle建立连续通知的开放连接。这使我能够在基础Oracle表更改时自动刷新缓存的数据集。

数据缓存项可以由任意数量的WCF初始化线程访问,所有这些线程都在自己的事务范围内运行。我想你可以把它想象成将一个OracleConnection对象放在缓存中。更好的文本/示例代码块如下:

//beginning of a WCF service call
using (var tx = new TransactionScope())
{
    var conn = new OracleConnection();
    var cmd = new OracleCommand();
    // set up OCN on the cmd and connection
    var reader = cmd.ExecuteReader();
    cache.Add("conn", conn);
    cache.Add("cmd", cmd);
}

//beginning of a second wcf service call
using (var tx = new TransactionScope())
{
    var conn = cache.Get("conn");
    var cmd = cache.Get("cmd");
    var reader = cmd.ExecuteReader();
    // user reader to reload some data
}

重点是我的连接在多个线程和事务范围内具有较长的生命周期。

1 个答案:

答案 0 :(得分:5)

您是否尝试过允许您设置范围的TransactionScope构造函数之一?将范围设置为“Requires New”会为您的连接创建一个新事务以进行登记。将范围设置为“Suppress”会使您的连接不会在任何事务中登记。至少,这就是我如何阅读文档。我自己从未有过这种特殊需求。

using(var tx = new TransactionScope(TransactionScopeOption.RequiresNew))
{
    conn = new OracleConnection("my connection string");
    conn.Open();
}