EFCore如何更改跟踪状态
这里简单介绍下几种修改实体或者导航的跟踪状态。
1. 直接修改实体
对已在DbContext中跟踪的实体,直接操作,如给字段赋值、给导航属性赋值或者添加删除等
对未在DbContext中跟踪的实体,调用DbContext的Add、Update、Remove等方法,对已跟踪的实体也有效
需要注意所有添加的实体都需要调用DbContext.Add方法
2. 利用IStateManager.ChangingState(InternalEntityEntry,EntityState)
这个很简单,关键的是如何获得InternalEntityEntry,方法是调用IStateManager.TryGetEntry方法返回获得,该方法多个重载,主要都是传实体或者主键。
Type type = ....// 实体类型
DbContext ctx=...
var entityType = ctx.Model.FindEntityType(type);
var key = entityType.FindPrimaryKey();
var internalEntityEntry = ((IDbContextDependencies)ctx).StateManager.TryGetEntry(key,主键值);
((IDbContextDependencies)ctx).StateManager.ChangingState(internalEntityEntry,EntityState.Modified);
传主键同样可以获得隐式定义的ISkipNavigation的InternalEntityEntry,请看5中示例。
3. 利用集合导航属性的INavigationBase.GetCollectionAccessor()
INavigationBase.GetCollectionAccessor()返回IClrCollectionAccessor类型,使用它的Add和Remove方法操作集合,但是这里只针对已经在DbContext跟踪的才有效果,否则请参考5中的做法。
4. 利用IStateManager.InternalEntityEntryNotifier.NavigationCollectionChanged
对于已经跟踪的实体或导航,还可以NavigationCollectionChanged方法更改添加或者删除,主要是针对集合导航属性。
Object entity=...//实体对象,假设它有个集合导航属性:Data
var navigationEntry = ctx.Entry(entity).Navigation("Data");
var obj = entity.Data.First();//假设不为空
//将集合导航属性Data中obj修改成添加状态(EntityState.Added)
((IDbContextDependencies)ctx).StateManager.InternalEntityEntryNotifier.NavigationCollectionChanged(ctx.Entry(entity).GetInfrastructure(), navigationEntry.Metadata, new[] { obj }, Array.Empty<object>());
//将集合导航属性Data中obj修改成删除状态(EntityState.Deleted)
((IDbContextDependencies)ctx).StateManager.InternalEntityEntryNotifier.NavigationCollectionChanged(ctx.Entry(entity).GetInfrastructure(), navigationEntry.Metadata, Array.Empty<object>(), new[] { obj });
5. ISkipNavigation的状态修改
在第4点中,对于ISkipNavigation,无法修改导航的状态为EntityState.Unchanged,那么这里可以这么做。
if (navigationEntry.Metadata is ISkipNavigation skipNavigation)
{
var targetEntry = ((IDbContextDependencies)ctx).StateManager.GetOrCreateEntry(obj, navigationEntry.Metadata.TargetEntityType);
var key = skipNavigation.JoinEntityType.FindPrimaryKey();
var a = ctx.Entry(entity).GetInfrastructure()[skipNavigation.ForeignKey.PrincipalKey.Properties[0]];
var b = targetEntry[skipNavigation.Inverse.ForeignKey.PrincipalKey.Properties[0]];
var joinEntry = ((IDbContextDependencies)ctx).StateManager.TryGetEntry(key, new[] { a, b });
joinEntry.SetEntityState(EntityState.Unchanged);//也可以用IStateManager.ChangingState
}
如果ISkipNavigation还是未跟踪的,那么上面的代码joinEntry获得的是null,可以在上面代码之前加这两行就可以了。
((IDbContextDependencies)ctx).StateManager.CompleteAttachGraph()
((IDbContextDependencies)ctx).StateManager.InternalEntityEntryNotifier.NavigationCollectionChanged(ctx.Entry(entity).GetInfrastructure(), navigationEntry.Metadata, new[] { obj }, Array.Empty<object>());
作者:Rick Carter
出处:http://pains.cnblogs.com/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】博客园社区专享云产品让利特惠,阿里云新客6.5折上折
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· [翻译] 为什么 Tracebit 用 C# 开发
· Deepseek官网太卡,教你白嫖阿里云的Deepseek-R1满血版
· 2分钟学会 DeepSeek API,竟然比官方更好用!
· .NET 使用 DeepSeek R1 开发智能 AI 客户端
· 刚刚!百度搜索“换脑”引爆AI圈,正式接入DeepSeek R1满血版