Nea*_*alR 5 asp.net-mvc entity-framework razor asp.net-mvc-4
在我的ASP MVC 4站点遇到一个奇怪的问题.如果用户打开create表单并尝试将代理添加到我们的数据库,我们自然会首先在字段上运行验证.如果出现错误,我们会将代理保存为未完成状态,并将用户重定向回创建页面.
当用户尝试重新保存代理时出现错误.在第一个回发后,代理将保存到数据库中并生成PK.然而,在第二个回发后,PK被发送到服务器,其值0不是刚刚自动生成的值.
我HiddenFor在Create视图上添加了一个,但是0每次都会显示一个值.
我还逐步完成了代码,以确保在.Save调用之后生成PK,在调用时仍然存在,return View并且还确保Model.ID在呈现视图时属性包含相同的值.
无论如何,如果我右键单击页面并查看源,隐藏字段将呈现如下:
<input data-val="false" data-val-number="The field ID must be a number." data-val-required="The ID field is required." id="ID" name="ID" type="hidden" value="0" />
Run Code Online (Sandbox Code Playgroud)
模型
public partial class AgentTransmission
{
public int ID { get; set; }
.
.
.
}
Run Code Online (Sandbox Code Playgroud)
视图
@model MonetModelFromDb.Models.AgentTransmission
@{
ViewBag.Title = "Create new Agent";
}
@Html.HiddenFor(model => model.CreatedDate, new { data_val = "false" })
@Html.HiddenFor(model => model.CreatedOperator, new { data_val = "false" })
@Html.HiddenFor(model => model.ReferenceNumber, new { data_val = "false" })
@Html.HiddenFor(model => model.Region, new { data_val = "false" })
@Html.HiddenFor(model => model.INDDist, new { data_val = "false" })
@Html.HiddenFor(model => model.LastChangeDate, new { data_val = "false" })
@Html.HiddenFor(model => model.LastChangeOperator, new { data_val = "false" })
@Html.HiddenFor(model => model.EditTaxId, new { data_val = "false" })
@Html.HiddenFor(model => model.ParentId, new { data_val = "false" })
@Html.HiddenFor(model => model.IsSubstat, new { data_val = "false" })
@Html.HiddenFor(model => model.ID, new { data_val = "false" })
Run Code Online (Sandbox Code Playgroud)
渲染HiddenFor部分
<input data-val="false" data-val-date="The field CreatedDate must be a date." id="CreatedDate" name="CreatedDate" type="hidden" value="" />
<input data-val="false" id="CreatedOperator" name="CreatedOperator" type="hidden" value="" />
<input data-val="false" id="Region" name="Region" type="hidden" value="NM-834" />
<input data-val="false" id="INDDist" name="INDDist" type="hidden" value="834" />
<input data-val="false" data-val-date="The field LastChangeDate must be a date." data-val-required="The LastChangeDate field is required." id="LastChangeDate" name="LastChangeDate" type="hidden" value="4/8/2015 10:43:30 AM" />
<input data-val="false" id="LastChangeOperator" name="LastChangeOperator" type="hidden" value="TYPCLS" />
<input data-val="false" data-val-required="The EditTaxId field is required." id="EditTaxId" name="EditTaxId" type="hidden" value="False" />
<input data-val="false" data-val-number="The field ParentId must be a number." id="ParentId" name="ParentId" type="hidden" value="" />
<input data-val="false" data-val-required="The IsSubstat field is required." id="IsSubstat" name="IsSubstat" type="hidden" value="False" />
<input data-val="false" data-val-number="The field ID must be a number." data-val-required="The ID field is required." id="ID" name="ID" type="hidden" value="0" />
Run Code Online (Sandbox Code Playgroud)
调节器
[HttpPost]
[MonetAuthorize]
public ActionResult Create(AgentTransmission agenttransmission, bool andAddAgent = false)
{
.
.
.
//Determine if this is first POST or not
if (agenttransmission.ID > 0)
{
db.Entry(agenttransmission).State = EntityState.Modified;
}
else
{
db.AgentTransmission.Add(agenttransmission);
}
db.SaveChanges();
//Send back to view if errors pressent
if (!String.IsNullOrWhiteSpace(errorMsg))
{
return View(agenttransmission);
}
}
Run Code Online (Sandbox Code Playgroud)
编辑
如果我删除HiddenFor帮助器并简单地剪切/粘贴渲染的input标签,我就能捕获相关的PK值.但是,这有点hacky所以我希望找到一个更优雅的解决方案(如果可能的话)
<input data-val="false" id="ID" name="ID" type="hidden" value="@Model.ID" />
Run Code Online (Sandbox Code Playgroud)
这里经常提到这个问题的变化.基本上它归结为ModelState对象以及它的值覆盖视图的实际模型上的值的事实.发布表单时,0将在ModelState您的ID属性的对象中设置.在post操作中,您保存实体,这会导致其ID属性更新,但0仍在ModelState.当你返回视图时,再次0设置为值ID,因为,再次,这是什么ModelState.
它以这种方式工作的原因最好用一个例子来解释.假设您有一个表单,您正在编辑现有实体,该实体的Name属性设置为"Foo".用户在表单中将其更改为"Bar",然后发布表单.但是,他们忽略了填写必填字段,因此错误会阻止保存更新.此时会发生什么?如果我们使用模型值,该Name字段将重置为"Foo".但是,如果ModelState使用,则该Name字段保留用户对"Bar"的修改.在后一种情况下,他们只是修复错误并再次发布.在前者中,他们必须重新制作他们之前对表单所做的所有更改,这显然是一种非常糟糕的用户体验.
现在,至于如何解决这个问题.最好的方法是遵循PRG模式(Post-Redirect-Get).如果提交良好,并且您成功保存了更改,则不要返回视图,即使您希望用户能够立即进行其他更改.如果您需要,只需重定向回相同的操作,但重定向过程足以清除,ModelState以便用户现在与从数据库中提取的更新模型进行交互.
如果这不可行,那么你可以简单地清除ModelState.我建议不要完全清除它,因为你可能会引起一些非常真实的用户沮丧,如上例所示.如果你真的不能做重定向,那么试着只清除ModelState你真正需要的值.
ModelState.Remove("ID");
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
301 次 |
| 最近记录: |