Disposed时,TransactionScope抛出TransactionAbortedException

时间:2014-12-11 10:24:34

标签: c# .net transactions msdtc system.transactions

我有以下情况:

  • 父进程启动TransactionScope,使用TransactionInterop.GetTransmitterPropagationToken创建标识事务的标记,将数据插入数据库。 TransactionScope完成。
  • 启动另一个进程,使用上面提到的令牌创建Transaction,然后用于创建TransactionScope。此过程还会将数据插入数据库。 TransactionScope完成并处理。
  • 父进程尝试在此时处置其TransactionScope,并抛出TransactionAbortedException。例外情况并未提供任何无法提交的理由。

堆栈追踪:

at System.Transactions.TransactionStatePromotedAborted.BeginCommit(InternalTransaction tx, Boolean asyncCommit, AsyncCallback asyncCallback, Object asyncState)
at System.Transactions.CommittableTransaction.Commit()
at System.Transactions.TransactionScope.InternalDispose()
at System.Transactions.TransactionScope.Dispose()
at DistributedTransactions.Program.Main() in c:\Users\agolan.ALLSHARE\Documents\Visual Studio 2013\Projects\DistributedTransactions\DistributedTransactions\Program.cs:line 44
at System.AppDomain._nExecuteAssembly(RuntimeAssembly assembly, String[] args)
at System.AppDomain.ExecuteAssembly(String assemblyFile, Evidence assemblySecurity, String[] args)
at Microsoft.VisualStudio.HostingProcess.HostProc.RunUsersAssembly()
at System.Threading.ThreadHelper.ThreadStart_Context(Object state)
at System.Threading.ExecutionContext.RunInternal(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state)
at System.Threading.ThreadHelper.ThreadStart()

以下是该应用程序的代码:

internal class Program {
    private static void Main() {
        string connString = "data source=.;initial catalog=Test;integrated security=True;persist security info=True";
        string tokenFile = @"c:\Temp\token.txt";

        Transaction transaction = null;
        bool isChild = false;
        if (File.Exists(tokenFile)) {
            isChild = true;
            string tokenString = File.ReadAllText(tokenFile);
            byte[] token = Convert.FromBase64String(tokenString);
            transaction = TransactionInterop.GetTransactionFromTransmitterPropagationToken(token);
        }
        using (var transactionScope = transaction != null ? new TransactionScope(transaction) : new TransactionScope(TransactionScopeOption.Required, new TimeSpan(0, 15, 0))) {
            var curr = Transaction.Current;
            if (!isChild) {
                byte[] transactionBytes = TransactionInterop.GetTransmitterPropagationToken(curr);

                string tokenString = Convert.ToBase64String(transactionBytes);
                File.WriteAllText(tokenFile, tokenString);
            }
            using (var conn = new SqlConnection(connString)) {
                conn.Open();
                using (SqlCommand cmd = conn.CreateCommand()) {
                    Console.WriteLine("Enter id and value");
                    cmd.CommandText = "INSERT INTO KeyValue(Id, Value) VALUES (@1, @2)";
                    cmd.Parameters.Add(new SqlParameter("@1", Console.ReadLine()));
                    cmd.Parameters.Add(new SqlParameter("@2", Console.ReadLine()));
                    cmd.ExecuteNonQuery();
                }
            }
            transactionScope.Complete();

            Console.WriteLine("Dispose");
            Console.ReadLine();
        }
    }
}

问题:

  • 为什么不能提交?
  • 在这种情况下是否可以使用TransactionScope(我读回答Using transactions across processes但我想使用TransactionScope)?怎么样?

1 个答案:

答案 0 :(得分:2)

我能够在本地复制它,因为其他应用程序在父进程提交事务之前退出了。其他应用程序必须继续运行,直到父提交或回滚事务。一种方法是阻止应用程序退出,直到TransactionCompleted事件被触发。

我修改了您的代码以使用ManualResetEventSlim使子进程在退出之前等待父事务进程完成。

internal class Program
{
    private static void Main()
    {
        string connString = "data source=.;initial catalog=Test;integrated security=True;persist security info=True";
        string tokenFile = @"c:\Temp\token.txt";

Transaction transaction = null; bool isChild = false; if (File.Exists(tokenFile)) { isChild = true; string tokenString = File.ReadAllText(tokenFile); byte[] token = Convert.FromBase64String(tokenString); transaction = TransactionInterop.GetTransactionFromTransmitterPropagationToken(token); } using (var parentTxCompleteEvent = new ManualResetEventSlim(!isChild)) { using (var transactionScope = transaction != null ? new TransactionScope(transaction) : new TransactionScope(TransactionScopeOption.Required, new TimeSpan(0, 15, 0))) { var curr = Transaction.Current; if (!isChild) { byte[] transactionBytes = TransactionInterop.GetTransmitterPropagationToken(curr); string tokenString = Convert.ToBase64String(transactionBytes); File.WriteAllText(tokenFile, tokenString); } else { transaction.TransactionCompleted += (sender, e) => parentTxCompleteEvent.Set(); } using (var conn = new SqlConnection(connString)) { conn.Open(); using (SqlCommand cmd = conn.CreateCommand()) { Console.WriteLine("Enter id and value"); cmd.CommandText = "INSERT INTO KeyValue(Id, Value) VALUES (@1, @2)"; cmd.Parameters.Add(new SqlParameter("@1", Console.ReadLine())); cmd.Parameters.Add(new SqlParameter("@2", Console.ReadLine())); cmd.ExecuteNonQuery(); } } transactionScope.Complete(); Console.WriteLine("Dispose"); Console.ReadLine(); } parentTxCompleteEvent.Wait(); } }

}