EF7 beta6:保存多个实体时出错
EF7 beta6: Error saving multiple entities
我正在使用 ASP.NET5 和 Entity Framework 7.0.0-beta 6 创建一个 API,当我尝试在多个请求中执行各种更新时,我得到了这个异常:
无法跟踪 'Company',因为已跟踪具有相同密钥的此类型的另一个实例。对于新实体,请考虑使用 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() 或更改映射来解决它,但是如果我尝试在多个请求中更新多个实体,我得到异常:
- 第一个请求:更新 CompanyId 8
- 第一个请求:更新 CompanyId 9 !!错误
有人知道如何解决这个问题吗?
您收到此错误是因为您的 _repository
实例在内存中已有一个具有相同键的 Company
实例。当您跨线程重用 DbContext 的实例时,可能会发生这种情况。您上面的示例不包含有关 CompanyController._repository
如何实例化的代码。确保这不会在 HTTP 请求之间共享。
此外,在配置密钥之前添加此行。
entity.Property(c => c.CompanyId).ValueGeneratedOnAdd();
已解决: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>();
我也遇到了同样的问题。
我在 startup.cs 文件的 Configure 方法中将存储库注册为单例。将其更改为 AddScoped
解决了问题。
只需使用以下代码更新记录
_dbContext.Company.Update(company);
_dbContext.SaveChanges();
我正在使用 ASP.NET5 和 Entity Framework 7.0.0-beta 6 创建一个 API,当我尝试在多个请求中执行各种更新时,我得到了这个异常:
无法跟踪'Company',因为已跟踪具有相同密钥的此类型的另一个实例。对于新实体,请考虑使用 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() 或更改映射来解决它,但是如果我尝试在多个请求中更新多个实体,我得到异常:
- 第一个请求:更新 CompanyId 8
- 第一个请求:更新 CompanyId 9 !!错误
有人知道如何解决这个问题吗?
您收到此错误是因为您的 _repository
实例在内存中已有一个具有相同键的 Company
实例。当您跨线程重用 DbContext 的实例时,可能会发生这种情况。您上面的示例不包含有关 CompanyController._repository
如何实例化的代码。确保这不会在 HTTP 请求之间共享。
此外,在配置密钥之前添加此行。
entity.Property(c => c.CompanyId).ValueGeneratedOnAdd();
已解决: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>();
我也遇到了同样的问题。
我在 startup.cs 文件的 Configure 方法中将存储库注册为单例。将其更改为 AddScoped
解决了问题。
只需使用以下代码更新记录
_dbContext.Company.Update(company);
_dbContext.SaveChanges();