ASP.NET MVC3 ActionResult Edit没有';t将更改保存在数据库中
本文关键字:保存 存在 数据库 ActionResult MVC3 NET Edit 没有 ASP | 更新日期: 2023-09-27 18:27:43
我在控制器中编辑数据时遇到问题。我的两个型号是这样的:
[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();
}
我的问题是,如果我更改aktualisertZielgruppe.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>
从代码中很难判断,但看起来您是根据视图返回的id从数据库中加载对象,将其状态设置为Modified,然后保存同一对象。
在将对象保存到数据库之前,您实际上并没有将视图中的属性设置回对象。你在用什么ORM?大多数ORM在修改每个属性时都会处理对象的State,并在提交时将其自己的状态设置为modified。
在从数据库加载对象之前设置一个断点,快速观察视图中返回的值,然后在从数据库中加载后再次快速观察它们,我想你会发现你已经覆盖了它们
这是实体框架在跟踪更新的关联时出现问题的情况之一。
我假设你在这里没有做任何花哨的模型绑定(即,从数据库中提取模型绑定器)。
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");
}
}
还要注意,这种绑定到实体的方法会使您容易受到过度发布攻击。您可以通过在模型绑定过程中对模型属性进行白名单,或者对视图模型进行绑定,然后进行映射,来减轻这种攻击向量。
如果您像上面的例子一样传入FormCollection,您可以修改它的值,然后在调用TryUpdateModel()之前将FormCollection重新分配给ValueProvider,如下所示:
this.ValueProvider = collection.ToValueProvider();
然而,如果你正在传递一个对象,比如Zielgruppe,我不知道——试着自己弄清楚。
这是谷歌的第一个结果,所以我将在这里留下我的评论,以防我可以为其他人省去一些头疼的事情。
以下是我的情况。我已经创建了我的模型,并用脚手架创建了所有必要的视图。然后我又回去给我的模型添加了一把外键。
虽然索引和编辑视图列出了我定义的所有字段,但没有为外键赋值。这就产生了上下文。由于ModelState无效,永远不会执行SaveChanges()。
通过像下面这样的外键,问题得到了解决。
@Html.HiddFor(model=>model.ForeignKey)