entity-framework - 自动检测断开实体的更改
问题描述
我正在 Web 服务器上制作一个简单的编辑器,允许用户将数据更改/添加到存储在 MS SQL 服务器上的单个表中。
我正在使用 Entity Framework 6 来执行此操作,并且我想知道应该如何跟踪对实体模型所做的更改。
我希望我可以在上下文中加载新数据,并让上下文自动与数据库中的内容进行比较,然后调用 SaveChanges()。
但是从我在线阅读的内容来看,我似乎需要遍历所有数据,并检查自己发生了什么变化,以便我可以调用Context.Entry(myEntry).State = Added
或Context.Entry(myEntry).State = Modified
EF 是否无法自动检测新增内容、修改内容和未更改内容?
解决方案
我建议将 ViewModel 或 DTO 传递给视图,然后在提交时将它们映射回重新加载的实体。EF 只会自动更新设置值时更改的值。在不更改值的情况下设置值不会触发更新。(附加实体并设置其修改状态将更新所有列) 传递实体虽然方便,但比 UI 可能呈现的更多地暴露了您的数据结构,并且可以在发回之前被篡改。永远不要相信从客户那里回来的任何东西。当序列化到客户端时,数据不再是实体,而是 JSON 数据块。当发送回服务器时,它不是一个被跟踪的实体,它是一个带有实体签名的 POCO。EF 实体可以提供的任何更改跟踪都不会应用于客户端或在序列化/反序列化过程中继续存在。
例如:
给定一个有名字和出生日期的孩子。我们选择一个 DTO 传递给视图。视图更改了名称,我们取回 DTO 并将所有值,修改或以其他方式复制回实体并调用SaveChanges()
// For example, loading the child in the controller to pass to the view...
ChildDTO childDto = null;
using (var context = new TestDbContext())
{
childDto = context.Children
.Select(x => new ChildDto
{
ChildId = x.ChildId,
Name = x.Name,
BirthDte = x.BirthDate
}).Single(x => x.ChildId == 1);
}
// View updates just the name...
childDto.Name = "Luke";
// Example if the view passed DTO back to controller to update...
using (var context = new TestDbContext())
{
var child = context.Children.Single(x => x.ChildId == 1);
child.Name = childDto.Name;
child.BirthDate = childDto.BirthDate;
context.SaveChanges();
}
如果姓名更改而出生日期没有更改,则 EF 生成的更新语句只会更新姓名。如果实体名称已经是“Luke”,则不会发出更新语句。您可以使用 SQL 分析器验证此行为,以查看 SQL EF 是否/何时/什么发送到数据库。
Automapper 可以帮助简化此操作,以便将 DTO 重新放入实体中:
var mappingConfig = new MapperConfiguration(cfg =>
{
cfg.CreateMap<Child, ChildDTO>();
cfg.CreateMap<ChildDTO, Child>();
});
然后在阅读时,利用ProjectTo
而不是Select
:
using (var context = new TestDbContext())
{
childDto = context.Children
.ProjectTo<ChildDTO>(mappingConfig)
.Single(x => x.ChildId == 1);
}
...并且在更新实体时:
using (var context = new TestDbContext())
{
var child = context.Children.Single(x => x.ChildId == 1);
var mapper = mappingConfig.CreateMapper();
mapper.Map(childDto, child); // copies values from DTO to the entity instance.
context.SaveChanges();
}
在将值复制到实体之前验证 DTO 非常重要,无论是手动还是使用 Automapper。Automapper 配置也可以设置为仅复制预期/允许更改的值。
推荐阅读
- javascript - 有人可以将此箭头函数转换为普通函数吗?
- reactjs - 使用反应 cookie 运行测试时,道具“cookies”无效
- sms - Does Short message from CS/PS domain to a UE in same domain has to reach to the IP-SM-GW?
- c - How to delete a char image being called as a function in a loop in C?
- python - urllib.error.URLError:
我正在研究这个简单的程序来连接
import urllib.request import urllib.request as req import ssl req = urllib.request.Request('ftps://ftps-xyz.com') respon
- r - 错误消息:“要替换的项目数不是替换长度的倍数”错误,尽管向量长度相等
- html - 在第四个之后截断操作栏项目
- esp8266 - Can I use micropython in NodeMcu ?(model name: CH340 Lua WIFI)
- mysql - 如何在MYSQL中获取日期行差异
- vue.js - v-for 索引未显示在组件中