将当前事务传递给DbCommand

时间:2018-09-16 09:11:36

标签: c# asp.net-core-mvc ef-core-2.1

我正在使用ASP.NET Core 2.1和EF Core 2.1进行项目。尽管大多数查询和命令使用EF,但是某些单元仍需要直接调用存储过程。

我不能使用FromSql ,因为它需要基于实体模型的结果集。

假设我们有以下方法:

public Task CommandOne(DbContext context)
{
    Entity entity = new Entity
    {
        Name = "Name"
    };
    context.DbSet<Entity>().Add(entity);
    return context.SaveChangesAsync();
}

public async Task CommandTwo(DbContext context)
{
    DbCommand command = context.Database.GetDbConnection().CreateCommand();
    command.CommandText = "storedProcName";
    command.CommandType = System.Data.CommandType.StoredProcedure;

    using (var reader = await command.ExecuteReaderAsync().ConfigureAwait(false))
    {
        // read result sets
    }
}

如果我像这样在一个事务中调用两个命令:

public async Task UnitOfWork(DbContext dbContext)
{
    using (var transaction = await dbContext.Database.BeginTransactionAsync())
    {
        await CommandOne(dbContext);
        await CommandTwo(dbContext);
    }
}

此异常将发生:

  

BeginExecuteReader要求该命令在   分配给命令的连接处于未决的本地事务中。   该命令的Transaction属性尚未初始化。

我不得不提到,它不像command.Transaction = ...那样简单。这要求DbTransaction与EF使用的交易不同。

我已经库存了一个月! 有什么解决方法吗? 非常感谢。

1 个答案:

答案 0 :(得分:7)

  

我不得不提到,这不像command.Transaction = ....那样简单,这需要DbTransaction,这与EF所使用的交易有所不同。

实际上是。您只需要引用 Microsoft.EntityFrameworkCore.Relational 程序集并添加

using Microsoft.EntityFrameworkCore.Storage;

获得对GetDbTransaction扩展方法的访问权限:

if (context.Database.CurrentTransaction != null)
    command.Transaction = context.Database.CurrentTransaction.GetDbTransaction();