将ChangeTracker注入构造函数

时间:2018-10-02 11:18:43

标签: c# asp.net-core dependency-injection entity-framework-core

我想重用我们在另一个具有asp.net核心的项目中确立的概念。 IEntityMetadataProvider将调用注册的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以便注入?

1 个答案:

答案 0 :(得分:1)

如果我理解正确,则想在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);
            }
        }
    }
...