Entity Framework 实体更新时不更新外键

Entity Framework not updating foreign key when entity updates

我正在使用 Entity Framework 和 Web API 2. 我有一个船实体,具有名称、价格等属性。这些简单的属性在通过 Put 发送到网络时更新得很好 api 控制器。船实体还与名为 BoatType 的实体存在多对一关系。船型有"Yacht"、"Motor Yacht"等

在控制器中更新船实体时,数据库中船类型的外键不会更新。我是否必须以某种方式手动更新子实体值,或者有没有办法让 EF 自动执行此操作?

这是发送到网络的 PUT 请求示例 API:

{
  "$id":"1",
  "Images":[],
  "BoatType": {
    "$id":"3",
    "Boat":[],
    "Id":1,
    "DateCreated":"2015-09-15T13:14:39.077",
    "Name":"Yacht"
  },
  "Id":2,
  "Name":"Schooner",
  "Description":"A harmless schooner",
  "DateCreated":"2015-09-15T17:59:37.8",
  "Price":65000
}

这是网页版的更新功能API:

[ResponseType(typeof(void))]
public async Task<IHttpActionResult> Put(int id, Boat boat)
{
    if (id != boat.Id)
    {
        return BadRequest();
    }

    _db.Entry(boat).State = EntityState.Modified;

    try
    {
        await _db.SaveChangesAsync();
    }
    catch (DbUpdateConcurrencyException)
    {
        if (!BoatExists(id))
        {
            return NotFound();
        }
        else
        {
            throw;
        }
    }
    return StatusCode(HttpStatusCode.NoContent);
}

我看过类似的问题,例如 Entity Framework Code First Update Does Not Update Foreign Key, Entity Framework does not update Foreign Key object and Update foreign key using Entity Framework,但 none 似乎有完全相同的情况(或者答案没有帮助我理解我的问题)。

这是 Boat 和 BoatType 模型 类(由 EF 设计器自动生成)。

public partial class Boat
{
    [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", "CA2214:DoNotCallOverridableMethodsInConstructors")]
    public Boat()
    {
        this.Images = new HashSet<Image>();
    }

    public int Id { get; set; }
    public string Name { get; set; }
    public string Description { get; set; }
    public System.DateTime DateCreated { get; set; }
    public Nullable<double> Price { get; set; }

    [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", "CA2227:CollectionPropertiesShouldBeReadOnly")]
    public virtual ICollection<Image> Images { get; set; }
    public virtual BoatType BoatType { get; set; }
} 

public partial class BoatType
{
    [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", "CA2214:DoNotCallOverridableMethodsInConstructors")]
    public BoatType()
    {
        this.Boat = new HashSet<Boat>();
    }

    public int Id { get; set; }
    public System.DateTime DateCreated { get; set; }
    public string Name { get; set; }

    [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", "CA2227:CollectionPropertiesShouldBeReadOnly")]
    public virtual ICollection<Boat> Boat { get; set; }
}

好的,我知道问题出在哪里了。使用 SQL Server Profiler 查看 SQL Update 语句,我看到 Boat.BoatType 的外键甚至不在那里 - 所以我认为我的模型一定是在某个地方搞砸了。当我在设计器中创建模型时,我错误地将 Boat 和 BoatType 之间的关系设置为一对一。后来我意识到了这个错误,并将关联更改为一个 (BoatType) 到多个 (Boats),但这一定是在我生成数据库之后。哦! EF 处理关联的方式意味着仅更改图表中的关联类型是不够的 - 我当时应该 dropped/recreated 数据库约束。

因为我在数据库中只有测试数据,所以对我有用的是使用设计器中的 "Generate database from model..." 选项重新创建数据库。

一旦我让 PUT 正常工作,我必须解决的另一件事是 Web API报错"A referential integrity constraint violation occurred: The property value(s) of 'Boat.BoatTypeId' on one end of a relationship do not match the property value(s) of 'Boat.BoatType.Id' on the other end."。允许用户更改船型的select列表在客户端使用AngularJS绑定到Boat.BoatType。因此,在 PUT 数据中,Boat.BoatType 已更新为新值,但 Boat.BoatTypeId 未更改 - 因此出现 "referential integrity" 错误。所以我只是在发送 PUT 之前手动将 Boat.BoatTypeId 的值设置为 Boat.BoatType.Id,现在所有工作都按预期进行。