Несколько DbContexts в Приложении N-уровня
Я создаю свое первое приложение MVC N-уровня, и я столкнулся с дорожным блоком с тем, как управлять несколькими DbContexts
С моей базой данных первый подход.
у меня есть следующие слои
Presentation
Service (WCF)
Business
Data Access
мне не нужна ссылка на entity framework в моем слое сервиса, но я не вижу, как создать интерфейс или что-то для управления двумя контекстами. У меня он работает с одним контекстом, искаженным в IDatabaseFactory, но я не могу найти подход к управлению два.
ниже UnitOfWork
это создано в моем сервисе ctor, но каждый раз, когда я смотрю на него, я все еще привязан к SiteModelContainer
, когда на самом деле у меня другой контекст.
public class UnitOfWork : IUnitOfWork
{
private SiteModelContainer _context;
private readonly IDatabaseFactory _databaseFactory;
protected SiteModelContainer SiteContext
{
get { return _context ?? (_context = _databaseFactory.Get()); }
}
public UnitOfWork(IDatabaseFactory factory)
{
_databaseFactory = factory;
_context = _databaseFactory.Get();
}
//More code
}
public class DatabaseFactory : Disposable, IDatabaseFactory
{
private SiteModelContainer _dataContext;
public SiteModelContainer Get()
{
return _dataContext ?? (_dataContext = new SiteModelContainer());
}
protected override void DisposeCore()
{
if (_dataContext != null)
_dataContext.Dispose();
}
}
2 ответов
предоставление вашей фабрике и UnitOfWork параметра универсального типа может быть решением:
public class UnitOfWork<T> : IUnitOfWork<T>
where T : DbContext, new()
{
private T _context;
private readonly IDatabaseFactory<T> _databaseFactory;
protected T Context
{
get { return _context ?? (_context = _databaseFactory.Get()); }
}
public UnitOfWork(IDatabaseFactory<T> factory)
{
_databaseFactory = factory;
_context = _databaseFactory.Get();
}
//More code
}
public class DatabaseFactory<T> : Disposable, IDatabaseFactory<T>
where T : DbContext, new()
{
private T _dataContext;
public T Get()
{
return _dataContext ?? (_dataContext = new T());
}
protected override void DisposeCore()
{
if (_dataContext != null)
_dataContext.Dispose();
}
}
на IDatabaseFactory
и IUnitWork
интерфейсы также должны быть универсальными.
затем вы можете создать единицу работ для разных контекстов:
var factory1 = new DatabaseFactory<SiteModelContainer>();
var unitOfWork1 = new UnitOfWork<SiteModelContainer>(factory1);
var factory2 = new DatabaseFactory<AnotherModelContainer>();
var unitOfWork2 = new UnitOfWork<AnotherModelContainer>(factory2);
Edit:
чтобы избавиться от зависимости от EF в ваших классах обслуживания, вы можете попробовать что-то вроде этого. Служба знает только этих троих. интерфейсы:
public interface IUnitOfWorkFactory
{
IUnitOfWork Create(string contextType);
}
public interface IUnitOfWork : IDisposable
{
IRepository<TEntity> CreateGenericRepository<TEntity>()
where TEntity : class;
void Commit();
}
public interface IRepository<T>
{
IQueryable<T> Find(Expression<Func<T, bool>> predicate);
void Attach(T entity);
void Add(T entity);
// etc.
}
вот специальные реализации, специфичные для EF:
public class UnitOfWorkFactory : IUnitOfWorkFactory
{
public IUnitOfWork Create(string contextType)
{
switch (contextType)
{
case "SiteModelContainer":
return new UnitOfWork<SiteModelContainer>();
case "AnotherModelContainer":
return new UnitOfWork<AnotherModelContainer>();
}
throw new ArgumentException("Unknown contextType...");
}
}
public class UnitOfWork<TContext> : IUnitOfWork
where TContext : DbContext, new()
{
private TContext _dbContext;
public UnitOfWork()
{
_dbContext = new TContext();
}
public IRepository<TEntity> CreateGenericRepository<TEntity>()
where TEntity : class
{
return new Repository<TEntity>(_dbContext);
}
public void Commit()
{
_dbContext.SaveChanges();
}
public void Dispose()
{
_dbContext.Dispose();
}
}
public class Repository<T> : IRepository<T>
where T : class
{
private DbContext _dbContext;
private DbSet<T> _dbSet;
public Repository(DbContext dbContext)
{
_dbContext = dbContext;
_dbSet = dbContext.Set<T>();
}
public IQueryable<T> Find(Expression<Func<T, bool>> predicate)
{
return _dbSet.Where(predicate);
}
public void Attach(T entity)
{
_dbSet.Attach(entity);
}
public void Add(T entity)
{
_dbSet.Add(entity);
}
// etc.
}
ваша служба получит IUnitOfWorkFactory
injected:
public class MyService
{
private IUnitOfWorkFactory _factory;
public MyService(IUnitOfWorkFactory factory)
{
_factory = factory;
}
public MyMethod()
{
using(var unitOfWork1 = _factory.Create("SiteModelContainer"))
{
var repo1 = unitOfWork1.
CreateGenericRepository<SomeEntityTypeInSiteModel>();
// Do some work
unitOfWork1.Commit();
}
using(var unitOfWork2 = _factory.Create("AnotherModelContainer"))
{
var repo2 = unitOfWork2.
CreateGenericRepository<SomeEntityTypeInAnotherModel>();
// Do some work
unitOfWork2.Commit();
}
}
}
при создании службы вводится конкретный экземпляр фабрики:
var service = new MyService(new UnitOfWorkFactory());
имейте в виду, что тяжелая работа будет в абстрактном репозитории и его реализации. Как только у вас больше нет контекста EF в вашем классе обслуживания, вы должны имитировать множество методов в интерфейс repo поддерживает все необходимые сценарии для управления данными.
вы можете создать оболочку, которая является общим репозиторием для DbContexts (и использует базовый ObjectContext для поддержки этого).
вот пример, который я использовал в прошлом (который также отделяет ваш код от любой прямой зависимости от Entity Framework).
// Make your DbContext inherit from this. This goes in your Unit of Work.
public interface IEntitySetProvider : IDisposable
{
IEntitySet<T> CreateEntitySet<T>();
}
// This is your adapted DBContext
public class MyDbContext1 : DbContext, IEntitySetProvider
{
public IEntitySet<T> CreateEntitySet<T>()
{
return new EntitySet<T>(((IObjectContextAdapter)this).CreateObjectSet<T>());
}
.
.
.
}
/// <summary>
/// A wrapper for an IQueryable that exposes AddNew and Attach methods.
/// </summary>
/// <typeparam name = "T"></typeparam>
public interface IEntitySet<T> : IQueryable<T>
{
/// <summary>
/// Attaches the specified value and considers it new.
/// </summary>
/// <param name = "value">The value.</param>
void AddNew(T value);
/// <summary>
/// Attaches the specified value and considers it modified.
/// </summary>
/// <param name = "value">The value.</param>
void Attach(T value);
}
/// <summary>
/// An IEntitySet for Entity Framework.
/// </summary>
/// <typeparam name = "T"></typeparam>
internal class EntitySet<T> : IEntitySet<T> where T : class
{
private readonly ObjectSet<T> _objectSet;
public EntitySet(ObjectSet<T> objectSet)
{
_objectSet = objectSet;
}
#region IEntitySet<T> Members
public void AddNew(T value)
{
_objectSet.AddObject(value);
}
public void Attach(T value)
{
_objectSet.Attach(value);
_objectSet.Context.ObjectStateManager.ChangeObjectState(value, EntityState.Modified);
}
public IEnumerator<T> GetEnumerator()
{
return ((IQueryable<T>) _objectSet).GetEnumerator();
}
IEnumerator IEnumerable.GetEnumerator()
{
return ((IQueryable) _objectSet).GetEnumerator();
}
public Type ElementType
{
get { return ((IQueryable<T>) _objectSet).ElementType; }
}
public Expression Expression
{
get { return ((IQueryable<T>) _objectSet).Expression; }
}
public IQueryProvider Provider
{
get { return ((IQueryable<T>) _objectSet).Provider; }
}
#endregion
}