entity-framework - 使用通用 dbContext(多个 dbContext)获取通用存储库
问题描述
在我的 asp.net core 2 api 项目中使用具有多个上下文的实体框架构建通用存储库时,我迷路了。
我需要的是在不知道它使用哪个上下文的情况下将我的存储库注入控制器。
数据库上下文
public interface IDbContext
{
}
public interface IBlogContext : IDbContext
{
}
public interface IBlogLogContext : IDbContext
{
}
存储库
public class EfRepository<TEntity> : IRepository<TEntity> where TEntity : class, IEntity
{
private readonly IDbContext _context;
private readonly DbSet<TEntity> _dbSet;
public EfRepository(IDbContext context)
{
_context = context;
_dbSet = _context.Set<TEntity>();
}
/* other stuff */
}
启动
public void ConfigureServices(IServiceCollection services)
{
services.AddDbContext<BlogContext>(options =>
{
options.UseSqlServer(Configuration.GetConnectionString("DefaultConnection"))
.UseLoggerFactory(_logger);
});
services.AddDbContext<BlogLogContext>(options =>
{
options.UseSqlServer(Configuration.GetConnectionString("DefaultConnection"));
});
services.AddScoped(typeof(IRepository<>), typeof(EfRepository<>));
services.AddScoped<IBlogContext, BlogContext>();
services.AddScoped<IBlogLogContext, BlogLogContext>();
services.AddUnitOfWork<BlogContext, BlogLogContext>();
}
控制器
public class UserController : Controller
{
private readonly IRepository<User> _repository;
private readonly IAuthorService _authorService;
private readonly ILogger<UserController> _logger;
public UserController(IRepository<User> repository, IAuthorService authorService, ILogger<UserController> logger)
{
_repository = repository;
_authorService = authorService;
_logger = logger;
}
}
当我注入IRepository<User>
控制器构造函数时,.net core DI 如何解析它User
的BlogContext
实体,所以IDbContext
应该BlogContext
在我的EfRepository
实现中?
解决方案
我用这样的 DbContext 通用参数定义了我的 EfRepository:
public class EfRepository<TDbContext, TEntity> : IRepository<TEntity>
where TDbContext : DbContext
where TEntity : class, IEntity
{
public EfRepository(IDbContextProvider<TDbContext> dbContextProvider)
{
}
}
并得到这样的 DbContext-Entity 信息:
//EntityTypeInfo is a class has EntityType & DeclaringType props
private IList<EntityTypeInfo> GetEntityTypeInfos()
{
var result = new List<EntityTypeInfo>();
var dbContextTypes = this.GetType().Assembly.GetTypes().Where(t => (typeof(DbContext).IsAssignableFrom(t)));
foreach (var dbContextType in dbContextTypes)
{
result.AddRange(
from property in dbContextType.GetProperties(BindingFlags.Public | BindingFlags.Instance)
where
Common.IsAssignableToGenericType(property.PropertyType, typeof(DbSet<>)) &&
Common.IsAssignableToGenericType(property.PropertyType.GenericTypeArguments[0], typeof(IEntity))
select new EntityTypeInfo(
property.PropertyType.GenericTypeArguments[0],
property.DeclaringType
);
}
return result;
}
所以我可以使用 EntityTypeInfo 列表添加相关服务:
var entityTypeInfos = GetEntityTypeInfos();
foreach (var entityTypeInfo in entityTypeInfos)
{
//
//Add IDbContextProvider and DbContext services Here
//
//Add IRepository services like this
var repositoryInterface =typeof(IRepository<>);
var repositoryImplementation =typeof(EfRepository<,>);
var genericRepositoryType = repositoryInterface.MakeGenericType(entityTypeInfo.EntityType);
if (!Common.IocHasRegister(genericRepositoryType))
{
var implRepositoryType = repositoryImplementation.MakeGenericType(entityTypeInfo.DeclaringType, entityTypeInfo.EntityType);
services.AddScoped(genericRepositoryType, implRepositoryType);
}
}
推荐阅读
- python - method1() 缺少 1 个必需的位置参数:'self'
- angular - 角度:发生未处理的异常:找不到模块'webpack'?
- angular - Angular 9/10:根据类别显示产品数据
- database - 将文本输入数据库的不同选择
- python-3.x - 如何使用递归函数获得最深度的元组?
- shell - Shell awk - 从变量打印位置
- swift - Swift 将 StackView 隐藏到 TableView 中
- c# - .Net核心中带有DBcontext的连接字符串问题
- java - Chrome 不会自动打开网址
- networking - 重新启动后尝试启动时出现“代码:错误的 RIP 值”内核崩溃