首页 > 解决方案 > 如何检查 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
   }
})

标签: c#entity-framework-core.net-5change-tracking

解决方案


您要求的功能存在(例如,作为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 版本中进行更改时进行调整。如果您问我,与其浪费时间创建代码分析器来显示此类警告,不如公开这样的缺失和有用的功能。无论如何,它就是这样,我们应该使用他们给我们的东西。


推荐阅读