asp.net core 实现一个简单的仓储的方法
| 
                         一直有自己写个框架的想法,但是一直没有行动起来,最近比较闲,正好可以开工了. 现在已经完成了两部分.1.一个简单仓储,实现使用的是ef 2.IOC部分,这里是把内置的ioc替换成了aotofac,这部分感觉还是有一点缺陷的.下面说 仓储部分 这里主要是接口是实现,目前使用ef实现了仓储的接口.看一下代码 
 public interface IRepository<TEntity,TPrimaryKey>
  where TEntity : class
 {
  #region Select/Get/Query
  IQueryable<TEntity> GetAll();
  IQueryable<TEntity> GetAllIncluding(params Expression<Func<TEntity,object>>[] propertySelectors);
  List<TEntity> GetAllList();
  Task<List<TEntity>> GetAllListAsync();
  List<TEntity> GetAllList(Expression<Func<TEntity,bool>> predicate);
  Task<List<TEntity>> GetAllListAsync(Expression<Func<TEntity,bool>> predicate);
  T Query<T>(Func<IQueryable<TEntity>,T> queryMethod);
  TEntity Get(TPrimaryKey id);
  Task<TEntity> GetAsync(TPrimaryKey id);
  TEntity Single(Expression<Func<TEntity,bool>> predicate);
  Task<TEntity> SingleAsync(Expression<Func<TEntity,bool>> predicate);
  TEntity FirstOrDefault(TPrimaryKey id);
  Task<TEntity> FirstOrDefaultAsync(TPrimaryKey id);
  TEntity FirstOrDefault(Expression<Func<TEntity,bool>> predicate);
  Task<TEntity> FirstOrDefaultAsync(Expression<Func<TEntity,bool>> predicate);
  TEntity Load(TPrimaryKey id);
  #endregion
  #region Insert
  TEntity Insert(TEntity entity);
  Task<TEntity> InsertAsync(TEntity entity);
  #endregion
  #region Update
  TEntity Update(TEntity entity);
  Task<TEntity> UpdateAsync(TEntity entity);
  TEntity Update(TPrimaryKey id,Action<TEntity> updateAction);
  Task<TEntity> UpdateAsync(TPrimaryKey id,Func<TEntity,Task> updateAction);
  #endregion
  #region Delete
  void Delete(TEntity entity);
  Task DeleteAsync(TEntity entity);
  void Delete(TPrimaryKey id);
  Task DeleteAsync(TPrimaryKey id);
  void Delete(Expression<Func<TEntity,bool>> predicate);
  Task DeleteAsync(Expression<Func<TEntity,bool>> predicate);
  #endregion
  #region Aggregates
  int Count();
  Task<int> CountAsync();
  int Count(Expression<Func<TEntity,bool>> predicate);
  Task<int> CountAsync(Expression<Func<TEntity,bool>> predicate);
  long LongCount();
  Task<long> LongCountAsync();
  long LongCount(Expression<Func<TEntity,bool>> predicate);
  Task<long> LongCountAsync(Expression<Func<TEntity,bool>> predicate);
  #endregion
 }
下面是实现的部分代码,代码比较占版面,就不贴全了. 
 public abstract class RepositoryBase<TEntity,TPrimaryKey> : IRepository<TEntity,TPrimaryKey>
  where TEntity : class
 {
  public abstract IQueryable<TEntity> GetAll();
  public abstract IQueryable<TEntity> GetAllIncluding(params Expression<Func<TEntity,object>>[] propertySelectors);
  public virtual List<TEntity> GetAllList()
  {
   return GetAll().ToList();
  }
  public virtual async Task<List<TEntity>> GetAllListAsync()
  {
   return await Task.FromResult(GetAllList());
  }
 }
 public class EfRepositoryBase<TDbContext,TEntity,TPrimaryKey> : RepositoryBase<TEntity,TPrimaryKey>
  where TEntity : class
  where TDbContext : DbContext
 {
  public virtual TDbContext Context { private set; get; }
  public virtual DbSet<TEntity> Table => Context.Set<TEntity>();
  public EfRepositoryBase(TDbContext context)
  {
   Context = context;
  }
  public override IQueryable<TEntity> GetAll()
  {
   return Table;
  }
  public override IQueryable<TEntity> GetAllIncluding(params Expression<Func<TEntity,object>>[] propertySelectors)
  {
   if (propertySelectors == null)
   {
    return GetAll();
   }
   var linq = GetAll();
   foreach (var item in propertySelectors)
   {
    linq = linq.Include(item);
   }
   return linq;
  }
 }
注意看EfRepositoryBase继承了RepositoryBase,而RepositoryBase实现了IRepository.这里的RepositoryBase是所有实现的基类.GetAllList虚方法直接调用了抽象方法GetAll,这样在EfRepositoryBase中就可以减少很多代码了. 这里有个坑 EfRepositoryBase 是不能直接注册到IOC中的,因为EfRepositoryBase和IRepository的泛型参数个数不一致,ioc不能找到多出的一个泛型的值.使用仓储的时候继承EfRepositoryBase把dbcontext传进去就好了 
public class TestRepository<TEntity,TPrimaryKey> : EfRepositoryBase<TestContext,TPrimaryKey> where TEntity : class
{
 public TestRepository(TestContext context)
  : base(context)
 {
 }
}
IOC部分 asp.net core 微软提供了一个简单的IOC,但是接口比较少,替换成我们熟悉的ioc框架就方便多了. asp.net core 也有很方便的替换ioc的方法.简单说就是修改ConfigureServices方法的返回值为IServiceProvider.我使用了autofac,下面看代码. 
public IServiceProvider ConfigureServices(IServiceCollection services)
{
 services.AddMvc();
 return services.AddLuna<AutofacModule>();
}
public static IServiceProvider AddLuna<TModule>([NotNull]this IServiceCollection services)
 where TModule : IModule,new()
{
 var builder = new ContainerBuilder();
 builder.Populate(services);
 builder.RegisterModule<TModule>();
 return new AutofacServiceProvider(builder.Build());
}
public class AutofacModule : Module
{
 protected override void Load(ContainerBuilder builder)
 {
  builder.RegisterType<TestContext>();
  builder.RegisterGeneric(typeof(TestRepository<,>)).As(typeof(IRepository<,>))
   .InstancePerLifetimeScope();
 }
}
这里的Module和IModule是autofac的,功能已经实现了,但是作为框架来说直接暴露了autofac的东西显然是不合适的,下一步要实现一个框架自身的模块加载方式. 以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程小技巧。 (编辑:莱芜站长网) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!  | 
                  
- 用AJAX实现的无刷新的分页实现代码(asp.net)
 - 扩展ASP.NET数据缓存以在Web场之间共享
 - asp.net 汉字转换拼音及首字母实现代码
 - asp.net-mvc – ActionResult上的自定义属性
 - asp.net – 如何在Visual Studio中添加NUnit
 - asp.net-mvc – DotNetOpenAuth在ASP.NET MVC中的OAuth
 - asp.net-mvc-4 – 在EF迁移配置类的Seed方法中获取App_Data
 - ASP.NET AJAX中的$create函数是什么?
 - ASP.net WebAPI跨域调用问题的解决方法
 - asp.net确保javascript只加载一次
 
- global-variables – MVC 3 Razor _ViewStart中的
 - asp.net-mvc – 使用CORS在WebAPI中将text / pla
 - asp.net – Silverlight初始化错误2110 Internet
 - asp.net-mvc – ASP.NET MVC查看引擎解析顺序
 - 解决asp.net Sharepoint无法连接发布自定义字符串
 - 使用ASP.NET Web API 2.1配置依赖注入
 - asp.net Ajax之无刷新评论介绍
 - ASP.NET中的超链接控件和链接控件有什么区别?
 - .NET WebApi jsonapi.org支持
 - asp.net-mvc-3 – 在MVC Razor View中使用If语句
 
