在实体框架核心中使用一对多关系更新连接的实体

问题描述 投票:0回答:1

我有一对多的关系,我试图更新,但得到错误

数据库操作预计会影响1行但实际上会影响0行。自加载实体以来,数据可能已被修改或删除。有关理解和处理乐观并发异常的信息,请参阅http://go.microsoft.com/fwlink/?LinkId=527962

更新

只要受益人集合未被更改或更新,更新方法就会起作用。

代码看起来像

public class Employee
{
    public int Id { get; set; }
    public string Name { get; set; }

    private readonly List<Beneficiary> _beneficiaries;
    public IEnumerable<Beneficiary> Beneficiaries => _beneficiaries;

    public void AddBeneficiary(string name)
    {
        var beneficiary = new Beneficiary(Id, name);
        _beneficiaries.Add(beneficiary);
    }
}

public sealed class Beneficiary
{
    public int Id { get; set; }
    public int EmployeeId { get; private set; }
    public string Name { get; private set; }

    public Beneficiary(int employeeId, string name)
    {
        Name = name;
        EmployeeId = employeeId;
    }
}

//Extracted from repo method
public void Update(TEntity entity)
{
    if (entity != null)
        _dbSet.Update(entity);
}

// Extracted Actual update from application service
if (incomingEmployee.Beneficiaries.Any())
{
    if (employeeFromStore.Beneficiaries.Any())
    {
        employeeFromStore.Beneficiaries.ToList()
            .ForEach(existingBeneficiary => 
        employeeFromStore.RemoveBeneficiary(existingBeneficiary)); //Method skipped for brevity

        //Tried calling this too
        //_employeeRepository.UnitOfWork.SaveChanges();
    }

    incomingEmployee.Beneficiaries.ToList().ForEach(beneficiary =>
    {
        employeeFromStore.AddBeneficiary(beneficiary.Name);
    });
}

_employeeRepository.Update(employeeFromStore);
_employeeRepository.UnitOfWork.SaveChanges();

我觉得问题在于跟踪器如何处理导航属性集合的更新,但我可能是错的。

我还尝试使用下面的方法使用更改跟踪器进行更新

public void ApplyCurrentValues<TEntity>(TEntity original, TEntity current)
        where TEntity : class
{
    //if it is not attached, attach original and set current values
    base.Entry<TEntity>(original).CurrentValues.SetValues(current);
}

有没有人知道如何去做?

c# entity-framework-core
1个回答
0
投票

在阅读有关更新如何工作之后,我终于取得了进展。基本上提到here,和这issue。调用Update方法时,整个实体的状态(包括通过导航属性的所有可到达实体)都标记为已修改。如果您尝试将受益人添加到现有员工,则SaveChanges()会抛出,因为它希望使用提供的主键修改记录但在数据库中找不到。更新方法似乎不是最佳候选人。

我选择在EmployeeRepository中覆盖存储库基础的Update方法,如下所示

//Update method in the repository base
//Made method virtual to enable overriding
public virtual void Update(TEntity entity)
{
    if (entity != null)
        _dbSet.Update(entity);
}

//Update method from EmployeeRepository
public override void Update(Employee employee)
{
    var employeeFromStore = _unitOfWork.Employees.Where(p => p.Id == employee.Id)
        .Include(p => p.Beneficiaries)
        .SingleOrDefault();

    if (employeeFromStore != null)
    {
        context.Entry(employeeFromStore).CurrentValues.SetValues(employee);

        //Replacing the whole collection
        foreach (var beneficiary in employeeFromStore.Beneficiaries)
        {
            if (employee.Beneficiaries.Any())
                context.Entry(beneficiary).State = EntityState.Deleted;
        }

        //Adding new collection
        foreach (var beneficiaryToAdd in employee.Beneficiaries)
        {
            var newBeneficiary = new Beneficiary(beneficiaryToAdd.EmployeeId, beneficiaryToAdd.Name);

            employeeFromStore.AddBeneficiary(newBeneficiary.Name);
        }
    }
}

我相信这个答案可以改进或调整,以适应特定的情况。例如,除了清除持久收集之外,可以选择更新

© www.soinside.com 2019 - 2024. All rights reserved.