Asp.net mvc 使用实体框架检测控制器中模型属性的状态更改
我有一个大致标准的模型:Asp.net mvc 使用实体框架检测控制器中模型属性的状态更改,asp.net-mvc,asp.net-mvc-3,entity-framework,Asp.net Mvc,Asp.net Mvc 3,Entity Framework,我有一个大致标准的模型: public class Project { public int ID { get; set; } //... some more properties public DateTime StartDate { get; set; } public int Duration { get; set; } } 如果用户修改开始日期或项目持续时间,我必须调用函数来更新模拟。为了实现这一点,我想检测控制器中StartDate和Duration字段的状态变化
public class Project {
public int ID { get; set; }
//... some more properties
public DateTime StartDate { get; set; }
public int Duration { get; set; }
}
如果用户修改开始日期
或项目持续时间
,我必须调用函数来更新模拟。为了实现这一点,我想检测控制器中StartDate
和Duration
字段的状态变化
诸如此类:
if(project.StartDate.stateChange() || project.Duration.stateChange())
以下是控制器方法的示例:
[HttpPost]
public ActionResult Edit(Project project)
{
if (ModelState.IsValid)
{
if(project.StartDate.stateChange() || project.Duration.stateChange())
doSomething();
db.Entry(project).State = EntityState.Modified;
db.SaveChanges();
return RedirectToAction("Index");
}
return View(project);
}
有什么想法吗?我怎么能做到这一点呢?如果你想在不查询持久层的情况下做到这一点,我想在模型中添加旧值作为字段,然后将它们作为隐藏字段保留在页面中是最简单的解决方法 因此,在模型中添加CurrentStartDate和CurrentDuration:
public class Project {
public int ID { get; set; }
//... some more properties
public DateTime StartDate { get; set; }
public int Duration { get; set; }
public DateTime CurrentStartDate { get; set; }
public int CurrentDuration { get; set; }
}
然后在视图中添加带有旧值的隐藏字段:
@Html.HiddenFor(model => model.CurrentStartDate )
@Html.HiddenFor(model => model.CurrentDuration )
这将使您能够在控制器操作中比较所选值。我相信您可以将编辑的实体与从数据库读取的原始实体进行比较 比如:
public ActionResult Edit(Project project)
{
if (ModelState.IsValid)
{
var original = db.Find(project.ID);
bool changed = original.StartDate != project.StartDate || original.Duration != project.Duration;
if (changed)
{
original.StartDate = project.StartDate;
original.Duration = project.Duration;
doSomething();
db.Entry(original).CurrentValues.SetValues(project);
db.SaveChanges();
}
}
return View(project);
}
您可以通过ViewBag携带旧值来解决此问题 在行动中:
public ActionResult Edit(int? id)
{
//...Your Code
ViewBag.OrigStartDate = project.StartDate;
ViewBag.OrigDuration = project.Duration;
return View(project);
}
向视图中添加隐藏元素
...
@Html.Hidden("OrigStartDate", (DateTime)ViewBag.OrigStartDate)
@Html.Hidden("OrigDuration", (int)ViewBag.OrigDuration)
...
将这些参数添加到post方法中,并检查它们是否有更改
[HttpPost]
public ActionResult Edit(DateTime OrigStartDate, int OrigDuration)
{
if (ModelState.IsValid)
{
if (OrigStartDate != project.StartDate || OrigDuration != project.Duration)
doSomething();
db.Entry(project).State = EntityState.Modified;
db.SaveChanges();
return RedirectToAction("Index");
}
ViewBag.FileTypeId = new SelectList(db.FileTypes, "Id", "TypeName", dbinfo.FileTypeId);
return View(project);
}
使用.AsNoTracking().FirstOrDefault检索原始对象以进行比较:
public ActionResult Edit(Project project)
{
if (ModelState.IsValid)
{
Project original = db.AsNoTracking().FirstOrDefault( p => p.ID == project.ID);
if (original.StartDate != project.StartDate || original.Duration != project.Duration)
doSomething();
db.Entry(project).State = EntityState.Modified;
db.SaveChanges();
}
return View(project);
}
实际上,如果可能的话,我更愿意在控制器内进行所有检查,以保持代码更干净。你能详细介绍一下如何查询你的持久层吗,这听起来像是我试图实现的。然后你必须从数据库中获取当前项目,检查旧值,并与从客户端获取的值进行比较。我已经尝试过了:我得到以下异常
ObjectStateManager中已经存在具有相同键的对象。ObjectStateManager无法使用同一密钥跟踪多个对象。
我已经尝试过了。我得到以下错误:` ObjectStateManager中已经存在具有相同键的对象。ObjectStateManager无法跟踪具有相同密钥的多个对象。描述:执行当前web请求期间发生未处理的异常。请查看堆栈跟踪以了解有关错误的更多信息以及错误在代码中的起源。异常详细信息:System.InvalidOperationException:ObjectStateManager中已存在具有相同密钥的对象。ObjectStateManager无法跟踪具有同一密钥的多个对象。`带或不带'db.Entry(original).State=EntityState.Modified;'?好的,我站在更正,这似乎是工作,但必须手动分配original.value=project.value
可能会有点混乱,这可以通过替换db.Entry(原始).State=EntityState.Modified来解决代码>带有db.Entry(原始).CurrentValues.SetValues(项目)代码>@Chopo87请尊重本网站的其他用户。如果您在答案中发现任何错误,请编辑并更正。不要只在评论中留下“更新”,因为这里的评论是二等公民,很容易被遗漏。始终尽可能保持答案的准确性。谢谢。我们是否可以将自动映射器
与viewmodel一起使用,并使其更简单,而不是编写多个|
?我更喜欢此解决方案,因为它不涉及到数据库的额外往返