ASP.NET MVC3 ActionResult Edit不保存数据库中的更改

时间:2012-03-13 14:02:29

标签: c# asp.net-mvc-3 drop-down-menu edit viewbag

我在控制器中编辑数据时遇到问题。我的两个模型看起来像这样:

[Table("Zielgruppen")]
public class Zielgruppe
{
    public int Id { get; set; }
    public string Zielgruppenname { get; set; }
    public Bezug Bezug { get; set; }
}

public class Bezug
{
    public int Id { get; set; }
    public string Bezugsname { get; set; }
}

我的控制器中的功能是:

public ActionResult Edit(int id)
{
    Zielgruppe zielgruppe = _db.Zielgruppe.Include("Bezug").Single(z => z.Id == id);
    ViewBag.BezugsId = new SelectList(_db.Bezug, "Id", "Bezugsname", zielgruppe.Bezug.Id);
    return View(zielgruppe);
}

[HttpPost]
public ActionResult Edit(Zielgruppe aktualisierteZielgruppe)
{
    if(ModelState.IsValid)
    {
        aktualisierteZielgruppe.Bezug = _db.Bezug.Find(aktualisierteZielgruppe.Bezug.Id);
        _db.Entry(aktualisierteZielgruppe).State = EntityState.Modified;
        _db.SaveChanges();
        return RedirectToAction("Index");
    }
    ViewBag.BezugsId = new SelectList(_db.Bezug, "Id", "Bezugsname", aktualisierteZielgruppe.Bezug.Id);
    return View();            
}

我的问题是,如果我更改了aktualisierteZielgruppe.Bezug,则更改将不会保存在数据库中。

这是我的edit.cshtml:

@model Medien_Archiv.Models.Zielgruppe

@{
    ViewBag.Title = "Edit";
}

<h2>Edit</h2>

<script src="@Url.Content("~/Scripts/jquery.validate.min.js")" type="text/javascript"></script>
<script src="@Url.Content("~/Scripts/jquery.validate.unobtrusive.min.js")" type="text/javascript"></script>

@using (Html.BeginForm()) {
    @Html.ValidationSummary(true)
    <fieldset>
        <legend>Zielgruppe</legend>

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

        <div class="editor-label">
            @Html.LabelFor(model => model.Zielgruppenname)
        </div>
        <div class="editor-field">
            @Html.EditorFor(model => model.Zielgruppenname)
            @Html.ValidationMessageFor(model => model.Zielgruppenname)
        </div>
        <div class="editor-label">
            @Html.LabelFor(model => model.Bezug)
        </div>
        <div class="editor-field">
            @Html.DropDownListFor(model => model.Bezug.Id, ViewBag.BezugsId as SelectList)   
        </div>
        <p>
            <input type="submit" value="Save" />
        </p>
    </fieldset>
}

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

4 个答案:

答案 0 :(得分:1)

很难从您的代码中判断,但看起来您正在根据从视图返回的ID从数据库加载对象,将其设置为Modified,然后保存相同的对象。

在将属性保存到数据库之前,看起来您实际上并未将视图中的属性设置回对象。您使用的是什么ORM?大多数ORM将在修改每个属性时处理对象的状态,并且在提交时基本上将其自己的状态设置为Modified。

在从数据库加载对象之前设置断点,快速查看从视图返回的值,然后在从数据库加载后再次快速监视它们,我想你会发现你已经覆盖了它们< / p>

答案 1 :(得分:0)

这是实体框架动摇的情况之一 - 跟踪更新的关联。

我假设你没有在这里做任何花哨的模型绑定(即模型绑定器从数据库中拉出来)。

if(ModelState.IsValid)
{
    aktualisierteZielgruppe.Bezug = _db.Bezug.Find(aktualisierteZielgruppe.Bezug.Id);

    // The problem is here.  EF doesn't mark associations as modified
    // the same way it tracks scalar types.
    _db.Entry(aktualisierteZielgruppe).State = EntityState.Modified;
    _db.SaveChanges();
    return RedirectToAction("Index");
}

这是更传统的方式:

[HttpPost]
public ActionResult Edit(int id, FormCollection collection)
{
    // Get the original record to edit from the database.
    var gruppe = _db.Zielgruppe.Include("Bezug").Single(z => z.Id == id);

    // This will attempt to do the model binding and map all the submitted 
    // properties to the attached entity.
    if (TryUpdateModel(grupppe))
    {
        _db.SaveChanges();
        return RedirectToAction("Index");
    }
}

另请注意,这种绑定到实体的方法会让您open to over-posting attacks.您可以通过在模型绑定期间列出模型属性来缓解此攻击向量,或者绑定到视图模型然后映射。

答案 2 :(得分:0)

如果您像上面的示例之一一样传入FormCollection,则可以修改其值,然后在调用TryUpdateModel()之前将FormCollection重新分配给ValueProvider,如下所示:

this.ValueProvider = collection.ToValueProvider();

然而,如果你传递一个像Zielgruppe这样的对象,我不知道 - 试图自己想出那个。

答案 3 :(得分:0)

这是谷歌的第一个结果,所以我在这里留下我的评论,以防我为别人省下一些头痛。

以下是我案件中发生的事情。我创建了我的模型并使用脚手架创建了所有必要的视图。然后我回去为我的模型添加了一个外键。

虽然索引和编辑视图列出了我所嵌入的所有字段,但没有为外键指定值。这导致context.SaveChanges()从未执行,因为ModelState无效。

通过下面的外键解决问题。

@ Html.HiddenFor(model =&gt; model。 ForeignKey