c# - 如何检查 DbContext 是否正在跟踪与 {'id'} 具有相同键值的实体?
问题描述
我认为这个问题是不言自明的:
如何检查是否正在跟踪与 {'id'} 具有相同键值的实体?
例如:
var blog = anotherDbContext.Blogs.Include(b => b.Posts).Find(...);
dbContext.ChangeTracker.AttachGraph(blog, node => {
if (node.Entry.State == EntityState.Detached) {
//how do I check, whether there is already an entity with the same key as node.Entity
node.Entry.State = EntityState.Unchanged; //this might throw InvalidOperationException
}
})
解决方案
您要求的功能存在(例如,作为Find
方法的一部分),但不幸的是没有公开(出于某种未知原因,他们认为这对他们没有用,并在发生异常时抛出异常)。
但是由于 EF Core 代码是开源的,因此不难提取一个自定义扩展方法,该方法使用与内部实现类似的方法Find
(确实可以搜索Local
的属性DbSet
,但使用非通用代码很难找到,效率低下,必须处理未知的关键属性名称和类型——所有这些都已由内部代码处理)。方法如下:
using Microsoft.EntityFrameworkCore.ChangeTracking;
using Microsoft.EntityFrameworkCore.ChangeTracking.Internal;
using Microsoft.EntityFrameworkCore.Infrastructure;
namespace Microsoft.EntityFrameworkCore
{
public static class ChangeTrackerExtensions
{
public static object FindTracked(this DbContext context, object entity)
{
var entityType = context.Model.FindRuntimeEntityType(entity.GetType());
var key = entityType.FindPrimaryKey();
var keyProperties = key.Properties;
var keyValues = new object[keyProperties.Count];
for (int i = 0; i < keyValues.Length; i++)
keyValues[i] = keyProperties[i].GetGetter().GetClrValue(entity);
var stateManager = context.GetService<IStateManager>();
return stateManager.TryGetEntry(key, keyValues)?.Entity;
}
}
}
唯一的问题是使用标记为“内部基础架构”一部分的代码,因此您会收到一些警告。只需忽略或禁止它们。上面的代码适用于 EF Core 3.x 和 EF Core 5.x , 并且可以/应该在未来的 EF Core 版本中进行更改时进行调整。如果您问我,与其浪费时间创建代码分析器来显示此类警告,不如公开这样的缺失和有用的功能。无论如何,它就是这样,我们应该使用他们给我们的东西。