DbUpdateExecption-保存不公开其关系的外键属性的实体时发生错误

时间:2013-06-05 12:00:55

标签: c# asp.net-mvc entity-framework asp.net-mvc-4 entity-framework-5

我的Asp.net Mvc4应用程序出了问题。

我设计了一个“用户”表和一个“可投影”,并且存在多对多的关系。

我的用户表:

 public partial class User
{
    public User()
    {
        this.Role = new HashSet<Role>();
        this.Project = new HashSet<Project>();
    }

    public int Id { get; set; }
    public string Username { get; set; }
    public string Password { get; set; }

    public virtual ICollection<Role> Role { get; set; }
    public virtual ICollection<Project> Project { get; set; }
}

我的项目表:

public partial class Project
{
    public Project()
    {
        this.Thresholds = new HashSet<Thresholds>();
        this.User = new HashSet<User>();
        this.Testrelease = new HashSet<Testrelease>();
    }

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

    public virtual ICollection<Thresholds> Thresholds { get; set; }
    public virtual ICollection<User> User { get; set; }
    public virtual ICollection<Testrelease> Testrelease { get; set; }
}

在我的项目控制器中:

public ActionResult EditProject(int id = 0)
    {
        Project project = db.Project.Find(id);
        if (project == null)
        {
            return HttpNotFound();
        }
        List<CheckBoxListInfoInt> userCheck = new List<CheckBoxListInfoInt>();
        foreach (User U in db.User.ToList())
        {
            userCheck.Add(new CheckBoxListInfoInt
            {
                ValueInt = U.Id,
                DisplayText = U.Username,
                IsChecked = (project.User.Contains(U))
            });
        }
        ViewBag.P_usrCB = userCheck;
        ViewData["pid"] = id;
        return View(project);
    }

    //
    // POST: /KPI_Data/Edit/5

    [HttpPost]
    [ValidateAntiForgeryToken]
    public ActionResult EditProject(Project project)
    {
        if (ModelState.IsValid)
        {
            db.Project.Attach(project);

            db.Entry(project).State = EntityState.Modified;

            if (project.User.Count > 0)
                project.User.Clear();


            List<string> usrlist = new List<string>();
            var ucb = Request.Form["P_usrCB"];
            if (ucb != null)
                foreach (string item in ucb.Split(','))
                {
                    int UserId = Convert.ToInt32(item);
                    User usr = db.User.Single(x => x.Id == UserId);
                    project.User.Add(usr);
                    usrlist.Add(usr.Username);
                }
            db.SaveChanges();
            return RedirectToAction("Index");
        }
        return View(project);
    }

错误发生在

  

db.SaveChanges();

在我的HttpPost ActionMethod

错误消息是:

  

保存不公开外键的实体时发生错误   他们关系的属性。 EntityEntries属性将   返回null,因为无法将单个实体标识为源   例外。可以在保存时处理异常   通过在实体类型中公开外键属性更容易。看到   InnerException以获取详细信息。

为什么会出现此错误?什么是解决方案? thx

1 个答案:

答案 0 :(得分:-1)

避免使用db.Entry(ptvm.place).State = EntityState.Modified;,因为它会导致并发冲突而不进行更新。将ViewModel用于多个关系而不是两个表。

您必须使用UpdateModel(table object, "model");

完整示例如下:

[HttpPost]
public ActionResult PlaceTag(PlacesWithTagsViewModel model)
{
    if (ModelState.IsValid)
    {
       tag tagtest = GetTagById(model.tagid);
       tag.name= model.tag.name;
       tag.nameplural = model.tag.nameplural;
       UpdateModel(tag, "model");
       db.SaveChanges();
       return RedirectToAction("Index", "Dashboard", new { id = 5 });
    } 
}

UpdateModel的优点是您只需要提及您更新的那些字段,避免那些保持静态的字段。通过这种方式,您可以在编辑视图中使用Viewmodel更新相关数据。