HTTP 上的 MVC 模型子对象为 null post

MVC Model child object null on HTTP post

希望有人能帮助我。我是 MVC 的新手,来自 winforms/console/vb6background.

很抱歉,如果这个问题已经得到解答,我正在努力了解如何解决以下问题。

我有一个视图模型:

public class testvm
{
    public int id { get; set; }
    public DateTime date { get; set; }
    public student studentID { get; set; }

    public testvm() { }

    public testvm (student s)
    {
        studentID = s;
    }
}

我在将此 ViewModel 传递给视图之前预填充它的学生子对象。

学生模型:

public class student
{
    [Key]
    public int ID { get; set; }
    public string Name { get; set; }
}

我遇到的问题是,当模型返回到创建 HTTP post 方法时,学生子对象为空。

控制器代码:

// GET: testvms/Create
public ActionResult Create(int sId)
{
    student a = db.students.Find(sId);

    testvm b = new testvm(a);

    return View(b);
}

[HttpPost]
[ValidateAntiForgeryToken]
public ActionResult Create([Bind(Include = "id,date,student")] testvm testvm)
{
    if (ModelState.IsValid)
    {
        db.testvws.Add(testvm);
        db.SaveChanges();
        return RedirectToAction("Index");
    }

    return View(testvm);
}

查看代码:

@model WebApplication2.Models.testvm

@{
ViewBag.Title = "Create";
}

<h2>Create</h2>


 @using (Html.BeginForm()) 
{
@Html.AntiForgeryToken()

<div class="form-horizontal">
    <h4>testvm</h4>
    <hr />


    @Html.HiddenFor(model => model.studentID.ID)

    @Html.ValidationSummary(true, "", new { @class = "text-danger" })
    <div class="form-group">
        @Html.LabelFor(model => model.date, htmlAttributes: new { @class = "control-label col-md-2" })
        <div class="col-md-10">
            @Html.EditorFor(model => model.date, new { htmlAttributes = new { @class = "form-control" } })
            @Html.ValidationMessageFor(model => model.date, "", new { @class = "text-danger" })
        </div>
    </div>

    <div class="form-group">
        <div class="col-md-offset-2 col-md-10">
            <input type="submit" value="Create" class="btn btn-default" />
        </div>
    </div>
</div>
}

<div>
@Html.ActionLink("Back to List", "Index")
</div>

@section Scripts {
@Scripts.Render("~/bundles/jqueryval")
}

视图上的模型对象填充了学生信息。当它传回 Create POST 控制器时,学生子对象为空!

有人可以告诉我哪里出错了或实现此目标的正确方法吗?

我的申请表有很多表格,都需要预先填写学生信息。每个学生都会有许多表格需要填写。

非常感谢, 罗布

您的 属性 名称称为 studentId(尽管标准 C# 属性 命名约定规定它应该称为 StudentId):

public student studentID { get; set; }

但是在您的 Bind 属性中,您似乎指定了一些 student 属性,这在您的视图模型中并不存在:

[Bind(Include = "id,date,student")]

所以您可能想从控制器操作中删除这个 Bind 属性:

[HttpPost]
[ValidateAntiForgeryToken]
public ActionResult Create(testvm testvm)
{
    ...
}

另请注意,您的表单中只有一个学生 ID 隐藏字段:

@Html.HiddenFor(model => model.studentID.ID)

您没有对应的学生姓名隐藏字段 属性,因此它永远不会回传到您的控制器操作。

您的属性 [Bind(Include = "id,date,student")] 应包括您要设置的属性的名称,student 不在您的模型中,但 studentID 是,它们必须匹配.

您不必明确指定要绑定到模型的所有字段名称,默认情况下它们将被绑定,除非您使用 [=15= 告诉活页夹不要绑定它].因此,就目前而言,我建议删除您的 Include 属性以简化维护,除非有重要的使用它的原因,并且只需确保您绑定的模型仅包含您需要的值。

其次,您必须确保从 view 中的表单发回的任何值都具有相同的参数名称并且结构与您要绑定的值相同请求模型。

这个:

@Html.HiddenFor(model => model.studentID.ID)

不等于:

@Html.HiddenFor(model => model.studentID)

对于域模型(在您的情况下为 testvm)中的每个 属性,您的视图(或用于 ID 或其他非用户的 HiddenFor)必须有一个 EditorFor 或 Input 元素(如 TextBoxFor 等)ui 数据)。在 MVC 中绑定嵌套模型可能会很痛苦,因为 DefaultModelBinder 可能无法绑定整个 object.However 更安全的方法是仅在视图上公开 required 属性,例如

@Html.HiddenFor(model => model.studentID.ID)
@Html.HiddenFor(model => model.studentID.Name)

及以后在控制器端

[HttpPost]
[ValidateAntiForgeryToken]
public ActionResult Create(testvm testvm)
{
  var originalobj=db.get //get fresh copy from data store
 originalobj.Name=testvm.Name;
//     ...other properties
//perform required operations on originalobj
}

您可以将 AutoMapper 用于此目的,因为

Mapper.CreateMap<testvm,testvm>();
originalobj=Mapper.Map<testvm,testvm>(testvm,originalobj);

您可以在以下位置找到有关 Automapper 的更多信息: https://github.com/AutoMapper/AutoMapper/wiki/Getting-started