【问题标题】:EF7 beta6: Error saving multiple entitiesEF7 beta6:保存多个实体时出错
【发布时间】:2015-07-13 20:29:33
【问题描述】:

我正在使用 ASP.NET5 和 Entity Framework 7.0.0-beta 6 创建一个 API,当我尝试在多个请求中执行各种更新时,我得到了这个异常:

无法跟踪“公司”,因为已在跟踪另一个具有相同键的此类实例。对于新实体,请考虑使用 IIdentityGenerator 生成唯一键值。

这是我的代码:

public class MrBellhopContext : DbContext
{

    public DbSet<Company> Company { get; set; }

    protected override void OnModelCreating(ModelBuilder modelBuilder)
    {
        modelBuilder.Entity<Company>(entity =>
        {
            entity.Key(c => c.CompanyId);

            entity.Index(c => c.Name);

            entity.Property(c => c.CompanyId).ValueGeneratedOnAdd();
        });

        modelBuilder.UseSqlServerIdentityColumns();

        base.OnModelCreating(modelBuilder);
    }

}


public class Company
{

    public int CompanyId { get; set; }

    public string Name { get; set; }

    public string Description { get; set; }

    public string Phone { get; set; }

    public string Email { get; set; }

    public short StatusId { get; set; }

}


public class CompanyRepository : ICompanyRepository
{

    MrBellhopContext _dbcontext;


    public async Task UpdateAsync(Company company)
    {
        _dbcontext.Update(company);
        await _dbcontext.SaveChangesAsync();
    }
}



[Route("api/[controller]")]
public class CompanyController : Controller
{


    [HttpPut]
    public async void UpdateAsync([FromBody] Company company)
    {
        if ((!ModelState.IsValid) || (company == null))
        {
            Context.Response.StatusCode = 400;
            return;
        }
        else
        {
            await _repository.UpdateAsync(company);
        }
    }

}

我试图通过删除 ValueGeneratedOnAdd()UseSqlServerIdentityColumns() 或更改映射来解决它,但如果我尝试在多个请求中更新多个实体,我获取异常:

  1. 第一个请求:更新 CompanyId 8
  2. 第一个请求:更新 CompanyId 9 !!错误

有人知道如何解决这个问题吗?

【问题讨论】:

  • 已尝试将[Key] 属性添加到主键属性,在您的情况下为ComapnyId
  • 该错误表明您有两个具有相同 ID 的 Company 实例。通过查看您的代码看起来不像这种情况,但我不能确定。你能告诉调用者 UpdateAsync 吗?
  • @Bart 是的,当然。我包括代码。
  • 我认为在执行第一个操作后,DBContext 中还有一些东西还活着;当我执行第二个时,失败

标签: c# asp.net-core entity-framework-core


【解决方案1】:

已解决:https://github.com/aspnet/EntityFramework/issues/2652

我将存储库添加为单例:

services.AddSigleton<Data.Interfaces.Company.ICompanyRepository,Data.Repositories.Company.CompanyRepository>();

这意味着所有请求都共享一个存储库实例。您应该将其减少到 Scoped,以便每个请求都有一个存储库实例。除了避免您遇到的问题之外,它还可以确保您最终不会得到一个巨大的上下文实例来跟踪内存中数据库中的所有数据。

解决:

services.AddScoped<Data.Interfaces.Company.ICompanyRepository,Data.Repositories.Company.CompanyRepository>();

【讨论】:

  • 这在 ASP 5 vNext 和 EF 7 中为我修复了这个问题
  • @natemcmaster 已经有一个答案,它正在解释问题并提供解决方案。除了为未来的读者提供您的解决方案之外,也许您可​​以相信他的回答。
【解决方案2】:

您收到此错误是因为您的 _repository 实例在内存中已经有一个具有相同密钥的 Company 实例。当您跨线程重用 DbContext 的实例时,可能会发生这种情况。您上面的示例不包含有关如何实例化 CompanyController._repository 的代码。确保这不是在 HTTP 请求之间共享的。

另外,请在配置密钥之前添加此行。

entity.Property(c => c.CompanyId).ValueGeneratedOnAdd();

【讨论】:

    【解决方案3】:

    即使我也面临同样的问题。

    我在 startup.cs 文件的配置方法中将存储库注册为单例。将其更改为 AddScoped 解决了这个问题。

    只需要使用下面的代码来更新记录

    _dbContext.Company.Update(company);
    _dbContext.SaveChanges();
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 2020-08-20
      • 2012-10-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2015-11-05
      相关资源
      最近更新 更多