使用代码首先mvc与多对多关系问题

时间:2017-12-26 07:47:13

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

我创建了两个Class和Teacher模型,我想使用代码第一种方法和两个模型来配置多对多的关系。 当我将信息添加到类和教师表中时,值不会使用多对多关系实体框架代码第一种方法插入到TeacherClasses表中。

下面是模型和控制器的代码。

这是班级模型

public class Class
{
    [Key]
    public int ClassId { get; set; }

    [Required]
    public string ClassName { get; set; }

    public virtual ICollection<Teacher> Teachers { get; set; }

}

这是教师模型

public class Teacher
{
 [Key]  
    public int TeacherId { get; set; }

    [Required]
    public string TeacherName { get; set; }

    [DataType(DataType.Date)]
    public DateTime JoiningDate { get; set; }
    public decimal Salary { get; set; }
    public string Email { get; set; }


    public virtual ICollection<Class> Classes { get; set; }


    [NotMapped]
    public virtual Class Class { get; set; }
    [NotMapped]
    public virtual ICollection<int> SelectedClassList { get; set; }

}

这是教师控制器

[HttpPost]
    [ValidateAntiForgeryToken]
    public ActionResult Create([Bind(Include = "TeacherId,TeacherName,JoiningDate,Salary,Email,Classes,Class,SelectedClassList")] Teacher teacher)
    {
        if (ModelState.IsValid)
        {
            db.Teachers.Add(teacher);
            db.SaveChanges();

            teacher.Classes = new Collection<Class>();

            foreach (var classId in teacher.SelectedClassList)
            {
                teacher.Classes.Add(new Class {ClassId = classId });
            }

            return RedirectToAction("Index");
        }
        return View(teacher);
    }

这是数据库表的图像 in database class, teacher and teacherclasses tables shown

database

这是教师观点

<div class="form-group">
    @Html.Label("Class List", htmlAttributes: new { @class = "control-label col-md-2" })
    <div class="col-md-10">
        @*@Html.ListBox("SelectedClassList", ViewData["ClassList"] as MultiSelectList)
        @Html.ValidationMessageFor(model => model.Classes, "", new { @class = "text-danger" })*@
        @Html.ListBoxFor(model => model.SelectedClassList, new MultiSelectList(ViewBag.ClassList, "value", "text"), new { htmlAttributes = new { @class = "form-control" } })
        @Html.ValidationMessageFor(model => model.SelectedClassList, "", new { @class = "text-danger" })

    </div>
</div>

1 个答案:

答案 0 :(得分:1)

根据您的说明以及您保存数据的方式,您似乎有两种可能的方案:

  1. 插入教师并将其连接到现有课程。
  2. 现有教师连接到现有课程。
  3. 无论是哪种规则,规则始终是:

    • 首先现有实体附加到上下文
    • 然后添加新实体

    所以第一个场景是:

    foreach (var classId in teacher.SelectedClassList)
    {
        var class = new Class { ClassId = classId }; // "stub" entity
        db.Classes.Attach(class);
        teacher.Classes.Add(class); // Assuming that teacher.Classes isn't null
    }
    db.Teachers.Add(teacher); // <= Add
    db.SaveChanges();
    

    第二种情况:

    db.Teachers.Attach(teacher); // <= Attach
    foreach (var classId in teacher.SelectedClassList)
    {
        var class = new Class { ClassId = classId }; // "stub" entity
        db.Classes.Attach(class);
        teacher.Classes.Add(class); // Assuming that teacher.Classes isn't null
    }
    db.SaveChanges();
    

    如您所见,您永远不需要从数据库中提取现有实体。为了创建关联,所有EF需要知道两端实体的ID值,因此使用存根实体是理想的方法。

    在第一个场景中,如果您在添加教师之前没有附加课程,那么课程也会被标记为Added,并且您将空类记录插入到数据库中(如果他们完全通过验证。