多次通话'DbContext已被处理错误'

时间:2017-02-04 00:44:38

标签: c# entity-framework automapper

我已经设置了一个简单getCustomers()方法的API。端点在第一次调用时返回数据,但在第二次调用时返回错误。

  

错误:由于DbContext已被处理,无法完成操作

错误是由CustomerService

上的return db.Customers...引起的
  

问题:为什么这会在第一次通话时起作用,但在第二次通话时失败。怎么解决这个问题?

可在此处找到GitHub Repo: https://github.com/ChaseHardin/MyBookStore

以下是代码的演练:

控制器:

[RoutePrefix("api/customers")]
public class CustomerController : ApiController
{
    private readonly CustomerService _service = new CustomerService();

    [HttpGet, Route("")]
    public virtual IHttpActionResult Get()
    {
        var customers = _service.GetCustomers();
        return Ok(new {customers});
    }
}

客户服务:

public class CustomerService : BaseService
{
    public List<CustomerViewModel> GetCustomers()
    {
        using (var db = Application.GetDatabaseInstance())
        {
            return db.Customers.Select(AutoMapper.Mapper.Map<CustomerViewModel>).ToList();
        }
    }
}

BaseService

public class BaseService
{
    public BaseService()
    {
        AutoMapperConfiguration();
    }

    public void AutoMapperConfiguration()
    {
        Assembly.GetExecutingAssembly()
                 .GetTypes()
                 .Where(x => x.IsClass && x.Namespace == "MyBookStore.Business.ViewModels")
                 .ForEach(x => System.Runtime.CompilerServices.RuntimeHelpers.RunClassConstructor(x.TypeHandle));

        AutoMapper.Mapper.CreateMap<bool, short?>().ConvertUsing(x => x ? (short)1 : (short)0);
        AutoMapper.Mapper.CreateMap<short, bool>().ConvertUsing(x => x == 1);

        AutoMapper.Mapper.CreateMap<bool, int?>().ConvertUsing(x => x ? 1 : 0);
        AutoMapper.Mapper.CreateMap<int?, bool>().ConvertUsing(x => x.HasValue && x.Value == 1);

        AutoMapper.Mapper.CreateMap<short, int>().ConvertUsing(x => (int)x);
        AutoMapper.Mapper.CreateMap<int, int?>().ConvertUsing(x => x);
    }
}

CustomerViewModel

public class CustomerViewModel
{
    static CustomerViewModel()
    {
        AutoMapper.Mapper.CreateMap<Customer, CustomerViewModel>().ReverseMap();
    }

    public Guid CustomerId { get; set; }
    public string FirstName { get; set; }
    public string LastName { get; set; }
}

上下文设置:

public class Application
{
    private static readonly MyBookStoreEntity Context = new MyBookStoreEntity();

    public static MyBookStoreEntity GetDatabaseInstance()
    {
        return Context;
    }
}

1 个答案:

答案 0 :(得分:3)

使用using块时:

using (var db = Application.GetDatabaseInstance())

正在使用的对象&#34;将在街区尽头处理。 (using基本上是try/finally的语法简写,其中finally块在对象上调用.Dispose()。)

你在&#34;使用&#34;是这个值:

private static readonly MyBookStoreEntity Context = new MyBookStoreEntity();

此值为static,因此每次调用时它都是MyBookStoreEntity的相同实例。但是当你第一次打电话时,你.Dispose()。因此,任何后续调用都将在已处置的对象上进行。

基本上,您已经发现了static数据库上下文是非常坏主意的原因之一。您仍然可以将数据库上下文封装到一个类似的方法中,但每次都让该方法返回一个新实例:

public static MyBookStoreEntity GetDatabaseInstance()
{
    return new MyBookStoreEntity();
}

或者,如果该方法此时并未真正提供任何好处,那么只需创建您需要的上下文:

using (var db = new MyBookStoreEntity())

创建数据库上下文并不是一项特别繁重的操作。但是当你不使用它们时保持它们是。 (并在不同的操作之间共享它们充满了危险。)一个好的经验法则是离散地定义您需要为给定的应用程序操作执行的数据库操作,并以紧密的代码创建/使用/配置您的数据库连接尽可能阻止这些行动。