首页 > 解决方案 > 将 ChangeTracker 注入构造函数

问题描述

我想重用我们在另一个带有 asp.net 核心的项目中提出的概念。在保存实体之前,注册IEntityMetadataProvider的 s 将被调用,代码如下:DbContext

public class SppdContext : DbContext
{
    private readonly IEnumerable<IEntityMetadataProvider> _entityMetadataProviders;

    public SppdContext(DbContextOptions<SppdContext> options, IEnumerable<IEntityMetadataProvider> entityMetadataProviders) : base(options)
    {
        _entityMetadataProviders = entityMetadataProviders;
    }

    public override int SaveChanges()
    {
        PrepareSaveChanges();
        return base.SaveChanges();
    }

    private void PrepareSaveChanges()
    {
        ChangeTracker.DetectChanges();
        SetModifierMetadataOnChangedEntities();
    }

    private void SetModifierMetadataOnChangedEntities()
    {
        foreach (var entityMetadataProvider in _entityMetadataProviders)
        {
            entityMetadataProvider.SetModifierMetadataOnChangedEntities();
        }
    }
}

-

public class EntityMetadataProvider : IEntityMetadataProvider
{
    private readonly ChangeTracker _changeTracker;

    public EntityMetadataProvider(ChangeTracker changeTracker)
    {
        _changeTracker = changeTracker;
    }

    public void SetModifierMetadataOnChangedEntities()
    {
        var entriesToSetModifier = _changeTracker.Entries<BaseEntity>().Where(e => HasToSetModifierMetadata(e.State)).ToList();

        if (entriesToSetModifier.Count > 0)
        {
            var saveDate = DateTime.UtcNow;

            foreach (var entryToSetModifier in entriesToSetModifier)
            {
                SetModifierMetadataProperties(entryToSetModifier, saveDate);
            }
        }
    }

    private static bool HasToSetModifierMetadata(EntityState state)
    {
        return state == EntityState.Added || state == EntityState.Modified;
    }

    private static void SetModifierMetadataProperties(EntityEntry<BaseEntity> entry, DateTime saveDate)
    {
        if (entry.State == EntityState.Added)
        {
            entry.Property(e => e.CreatedOn).CurrentValue = saveDate;
        }
        else
        {
            entry.Property(e => e.CreatedOn).IsModified = false;
        }
    }
}

DbContext正在像这样注册:

var connectionString = "Data Source=.\\SQLEXPRESS; Initial Catalog=Sppd; Integrated Security=true; MultipleActiveResultSets=True;";
services.AddDbContext<SppdContext>(options => options.UseSqlServer(connectionString));

通过这个实现,我在打开时得到以下异常var context = serviceScope.ServiceProvider.GetRequiredService<SppdContext>();

Unable to resolve service for type 'Microsoft.EntityFrameworkCore.ChangeTracking.ChangeTracker' while attempting to activate 'SppdDocs.Infrastructure.DbAccess.EntityMetadataProvider'.

我如何注册ChangeTracker以便可以注入?

标签: c#asp.net-coredependency-injectionentity-framework-core

解决方案


如果我理解正确,您想在 SppdContext 的更改跟踪器中查找更改。

您无法通过注入获得 changetracker。它由您的上下文实例拥有。您应该在调用方法而不是构造函数时传递 Changetracker。

在 SpddContext 类中你会得到

private void SetModifierMetadataOnChangedEntities()
{
    foreach (var entityMetadataProvider in _entityMetadataProviders)
    {
        entityMetadataProvider.SetModifierMetadataOnChangedEntities(ChangeTracker);
    }
}

在 EntityMetaDataProvider 中:

public class EntityMetadataProvider : IEntityMetadataProvider
{

    public EntityMetadataProvider()
    {       
    }

    public void SetModifierMetadataOnChangedEntities(ChangeTracker changeTracker)
    {
        var entriesToSetModifier = changeTracker.Entries<BaseEntity>().Where(e => 
HasToSetModifierMetadata(e.State)).ToList();

        if (entriesToSetModifier.Count > 0)
        {
            var saveDate = DateTime.UtcNow;

            foreach (var entryToSetModifier in entriesToSetModifier)
            {
                SetModifierMetadataProperties(entryToSetModifier, saveDate);
            }
        }
    }
...

推荐阅读