EF7 beta6:複数のエンティティの保存中にエラーが発生しました

asp.net-core c# entity-framework-core

質問

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()を削除するかマッピングを変更して解決しようとしましたが、いくつかの要求でいくつかのエンティティを更新しようとすると、例外が発生します。

  1. 最初のreq:Update CompanyId 8
  2. 最初のreq:Update CompanyId 9 !!エラー

誰もこの問題を解決する方法を知っていますか?

受け入れられた回答

解決済み: https : //github.com/aspnet/EntityFramework/issues/2652

私はリポジトリをシングルトンとして追加していました:

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

これは、すべての要求がリポジトリの1つのインスタンスを共有していることを意味します。これをScopedに減らして、リクエストごとに1つのリポジトリインスタンスを作成するようにしてください。あなたが当てている問題を回避するだけでなく、メモリ内のデータベースからすべてのデータを追跡する巨大なコンテキストインスタンスにならないようにします。

解決するには:

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

人気のある回答

_repositoryのインスタンスに、同じキーを持つメモリ内にCompanyインスタンスが既に存在するため、このエラーが発生しています。これは、スレッド間でDbContextのインスタンスを再利用するときに発生します。上のサンプルには、 CompanyController._repositoryインスタンス化方法に関するコードは含まれていません。これがHTTPリクエスト間で共有されていないことを確認してください。

また、キーを設定する前に、この行を挿入してください。

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


Related

ライセンスを受けた: CC-BY-SA with attribution
所属していない Stack Overflow
ライセンスを受けた: CC-BY-SA with attribution
所属していない Stack Overflow