为什么要使用泛型仓储?好处是?

前两章在autofac注入的时候,用的User类作为例子,写了增删改查四个接口,也就是仓储的GRUD。

当我们再添加一个实体(比如Student)时,StudentRepository跟UserRepository代码几乎一样的代码,重复量很大,为了减少冗余、提高工作效率,使用泛型仓储最好不过了

好处:

减少代码冗余

提高了开发人员的工作效率

提高对数据库访问的维护

一、泛型仓储接口和泛型仓储实现类

泛型仓储接口

在类库项目上右键->添加->新建文件夹,命名为Repository,存放泛型仓储类。在Repository文件夹下面新建 泛型仓储接口类:IRepository,如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;
using NetCoreWebApi.Repository.Dto; namespace NetCoreWebApi.Repository.Repository
{
public interface IRepository<T> where T : class
{
/// <summary>
/// 添加实体(单个)
/// </summary>
/// <param name="entity">实体对象</param>
int Add(T entity); /// <summary>
/// 批量插入实体(多个)
/// </summary>
/// <param name="list">实体列表</param>
int AddRange(List<T> list); /// <summary>
/// 删除实体(单个)
/// </summary>
/// <param name="entity"></param>
int Remove(T entity); /// <summary>
/// 批量删除实体(多个)
/// </summary>
/// <param name="list">实体列表</param>
int RemoveRange(List<T> list);
/// <summary>
/// 获取所有
/// </summary>
/// <returns></returns>
IQueryable<T> GetAll();
/// <summary>
/// 分页条件查询
/// </summary>
/// <typeparam name="TKey">排序类型</typeparam>
/// <param name="pageIndex">当前页</param>
/// <param name="pageSize">每页大小</param>
/// <param name="predicate">条件表达式</param>
/// <param name="isAsc">是否升序排列</param>
/// <param name="keySelector">排序表达式</param>
/// <returns></returns>
Page<T> SearchFor<TKey>(int pageIndex, int pageSize, Expression<Func<T, bool>> predicate,
bool isAsc, Expression<Func<T, TKey>> keySelector);
/// <summary>
/// 获取实体(主键)
/// </summary>
/// <param name="id">主键id</param>
/// <returns></returns>
T GetModelById(object id);
/// <summary>
/// 获取实体(条件)
/// </summary>
/// <param name="predicate">条件表达式</param>
/// <returns></returns>
T GetModel(Expression<Func<T, bool>> predicate);
/// <summary>
/// 查询记录数
/// </summary>
/// <param name="predicate">条件表达式</param>
/// <returns>记录数</returns>
int Count(Expression<Func<T, bool>> predicate);
/// <summary>
/// 是否存在
/// </summary>
/// <param name="anyLambda">查询表达式</param>
/// <returns>布尔值</returns>
bool Exist(Expression<Func<T, bool>> anyLambda);
}
}

泛型仓储实现类

在Repository文件夹下面新建 泛型仓储实现类:Repository,并继承IRepository,如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;
using Microsoft.EntityFrameworkCore;
using NetCoreWebApi.Model;
using NetCoreWebApi.Repository.Dto; namespace NetCoreWebApi.Repository.Repository
{
public class Repository<T> : IRepository<T> where T : class
{
private readonly MyDbContext _dbContext;
private DbSet<T> _entity; /// <summary>
/// 构造函数
/// </summary>
/// <param name="dbContext"></param>
public Repository(MyDbContext dbContext)
{
_dbContext = dbContext;
}
private DbSet<T> Entity => _entity ?? (_entity = _dbContext.Set<T>());
/// <summary>
/// 添加实体(单个)
/// </summary>
/// <param name="entity">实体对象</param>
public int Add(T entity)
{
Entity.Add(entity);
return _dbContext.SaveChanges();
}
/// <summary>
/// 批量插入实体(多个)
/// </summary>
/// <param name="list">实体列表</param>
public int AddRange(List<T> list)
{
Entity.AddRange(list);
return _dbContext.SaveChanges();
} /// <summary>
/// 删除实体(单个)
/// </summary>
/// <param name="entity"></param>
public int Remove(T entity)
{
Entity.Remove(entity);
return _dbContext.SaveChanges();
}
/// <summary>
/// 批量删除实体(多个)
/// </summary>
/// <param name="list">实体列表</param>
public int RemoveRange(List<T> list)
{
Entity.RemoveRange(list);
return _dbContext.SaveChanges();
}
/// <summary>
/// 获取所有
/// </summary>
/// <returns></returns>
public IQueryable<T> GetAll()
{
return Entity.AsQueryable().AsNoTracking();
}
/// <summary>
/// 条件查询
/// </summary>
/// <typeparam name="TKey">排序类型</typeparam>
/// <param name="pageIndex">当前页</param>
/// <param name="pageSize">每页大小</param>
/// <param name="isAsc">是否升序排列</param>
/// <param name="predicate">条件表达式</param>
/// <param name="keySelector">排序表达式</param>
/// <returns></returns>
public Page<T> SearchFor<TKey>(int pageIndex, int pageSize, Expression<Func<T, bool>> predicate, bool isAsc,
Expression<Func<T, TKey>> keySelector)
{
if (pageIndex <= || pageSize <= ) throw new Exception("pageIndex或pageSize不能小于等于0");
var page = new Page<T> { PageIndex = pageIndex, PageSize = pageSize };
var skip = (pageIndex - ) * pageSize;
var able = Entity.AsQueryable().AsNoTracking();
if (predicate == null)
{
var count = Entity.Count();
var query = isAsc
? able.OrderBy(keySelector).Skip(skip).Take(pageSize)
: able.OrderByDescending(keySelector).Skip(skip).Take(pageSize);
page.TotalRows = count;
page.LsList = query.ToList();
page.TotalPages = page.TotalRows / pageSize;
if (page.TotalRows % pageSize != ) page.TotalPages++;
}
else
{
var queryable = able.Where(predicate);
var count = queryable.Count();
var query = isAsc
? queryable.OrderBy(keySelector).Skip(skip).Take(pageSize)
: queryable.OrderByDescending(keySelector).Skip(skip).Take(pageSize);
page.TotalRows = count;
page.LsList = query.ToList();
page.TotalPages = page.TotalRows / pageSize;
if (page.TotalRows % pageSize != ) page.TotalPages++;
}
return page;
}
/// <summary>
/// 获取实体
/// </summary>
/// <param name="id">主键id</param>
/// <returns></returns>
public T GetModelById(object id)
{
return Entity.Find(id);
} /// <summary>
/// 获取实体(条件)
/// </summary>
/// <param name="predicate">条件表达式</param>
/// <returns></returns>
public T GetModel(Expression<Func<T, bool>> predicate)
{
return Entity.FirstOrDefault(predicate);
}
/// <summary>
/// 查询记录数
/// </summary>
/// <param name="predicate"></param>
/// <returns></returns>
public int Count(Expression<Func<T, bool>> predicate)
{
return predicate != null ? Entity.Where(predicate).Count() : Entity.Count();
}
/// <summary>
/// 是否存在
/// </summary>
/// <param name="anyLambda"></param>
/// <returns></returns>
public bool Exist(Expression<Func<T, bool>> anyLambda)
{
return Entity.Any(anyLambda);
}
}
}

分页Page类

using System.Collections.Generic;

namespace NetCoreWebApi.Repository.Dto
{
public class Page<T>
{
/// <summary>
/// 当前页
/// </summary>
public int PageIndex { get; set; }
/// <summary>
/// 总页数
/// </summary>
public int TotalPages { get; set; }
/// <summary>
/// 集合总数
/// </summary>
public int TotalRows { get; set; }
/// <summary>
/// 每页项数
/// </summary>
public int PageSize { get; set; }
/// <summary>
/// 集合
/// </summary>
public IList<T> LsList { get; set; }
}
}

二、仓储的泛型的依赖注入。

修改Startup.cs启动类中ConfigureServices方法

        public static IContainer ApplicationContainer { get; set; }
/// <summary>
/// //负责注入服务
/// </summary>
/// <param name="services"></param>
/// <returns></returns>
public IServiceProvider ConfigureServices(IServiceCollection services)
{
//获取数据库连接字符串
var connectionStr = Configuration.GetConnectionString("SqlServer");
services.AddDbContext<MyDbContext>
(options => options.UseSqlServer(connectionStr,
e => e.MigrationsAssembly("NetCoreWebApi.Model")));
services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2);
//初始化容器
var builder = new ContainerBuilder();
//管道寄居
builder.Populate(services);
//注册业务
builder.RegisterAssemblyTypes(Assembly.Load("NetCoreWebApi.Repository"), Assembly.Load("NetCoreWebApi.Repository"))
.Where(t => t.Name.EndsWith("Repository"))
.AsImplementedInterfaces();
//注册仓储,所有IRepository接口到Repository的映射
builder.RegisterGeneric(typeof(Repository<>))
//InstancePerDependency:默认模式,每次调用,都会重新实例化对象;每次请求都创建一个新的对象;
.As(typeof(IRepository<>)).InstancePerDependency();
//构造
ApplicationContainer = builder.Build();
//将AutoFac反馈到管道中
return new AutofacServiceProvider(ApplicationContainer);
}

三、测试

修改业务层---UserRepository

给泛型类指定ThUser,红色字体是主要更改部分。

using System.Collections.Generic;
using System.Linq;
using NetCoreWebApi.Model.Models;
using NetCoreWebApi.Repository.Interface;
using NetCoreWebApi.Repository.Repository; namespace NetCoreWebApi.Repository.Implement
{
/// <summary>
/// 业务处理
/// </summary>
public class UserRepository:IUserRepository
{
private readonly IRepository<TbUser> _userRepository; /// <summary>
/// 构造函数
/// </summary>
/// <param name="userRepository"></param>
public UserRepository(IRepository<TbUser> userRepository)
{
_userRepository = userRepository;
}
/// <summary>
/// 添加用户
/// </summary>
/// <param name="entity"></param>
/// <returns></returns>
public int Add(TbUser entity)
{
return _userRepository.Add(entity);
}
/// <summary>
/// 删除用户
/// </summary>
/// <param name="entity"></param>
/// <returns></returns>
public int Remove(TbUser entity)
{
return _userRepository.Remove(entity);
}
/// <summary>
/// 查询用户
/// </summary>
/// <returns></returns>
public IList<TbUser> GetAll()
{
return _userRepository.GetAll().ToList();
}
/// <summary>
/// 条件查询
/// </summary>
/// <param name="name">姓名</param>
/// <param name="isAsc">是否正序</param>
/// <param name="pageIndex">页码</param>
/// <param name="pageSize">每页大小</param>
/// <returns></returns>
public object SearchFor(string name, bool isAsc, int pageIndex, int pageSize)
{
//条件表达式
Expression<Func<TbUser, bool>> predicate = e => string.IsNullOrWhiteSpace(name) || e.UserName.Equals(name);
var data = _userRepository.SearchFor(pageIndex, pageSize, predicate, isAsc, e=>e.CreateTime);
return data;
}
}
}

多条件查询

稍微修改下条件查询接口

        /// <summary>
/// 条件查询
/// </summary>
/// <param name="name">姓名</param>
/// <param name="name1">姓名1</param>
/// <param name="isAsc">是否正序</param>
/// <param name="pageIndex">页码</param>
/// <param name="pageSize">每页大小</param>
/// <returns></returns>
public object SearchFor(string name, string name1, bool isAsc, int pageIndex, int pageSize)
{
//条件表达式
Expression<Func<TbUser, bool>> predicate = e => (string.IsNullOrWhiteSpace(name) || e.UserName.Equals(name))&&
(string.IsNullOrWhiteSpace(name1)||e.UserName.Equals(name1));
var data = _userRepository.SearchFor(pageIndex, pageSize, predicate, isAsc, e=>e.CreateTime);
return data;
}

运行项目执行接口,可以看到跟之前一样

如果有其他实体只需要改变传入的T就可以了,不需要再重新创建TEntityRepository

ASP.NET Core 2.2 WebApi 系列【六】泛型仓储模式的更多相关文章

  1. ASP.NET Core 2.2 WebApi 系列【九】使用SignalR (作者:tenghao510 ) 学习及内容补充

    原文地址:  ASP.NET Core 2.2 WebApi 系列[九]使用SignalR 今天,看到了大牛的这篇博文,  发了一下评论, 我很惊喜, 没想到他很快就回复了我,  而且通过QQ帮助了S ...

  2. ASP.NET Core 3.0 WebApi 系列【2】.Net Core 3.0+ CodeFirst + MySql 实现数据的迁移

    写在前面 在前一小节中,我们创建了基于RESFULL风格的服务.这个小节,尝试使用CodeFirst+MySql 的方式实现数据迁移. 一.开发环境 [1]运行环境:win10 家庭版 [2]开发工具 ...

  3. ASP.NET Core 3.0 WebApi 系列【1】创建ASP.NET Core WebApi 项目

    目录 写在前面 一.运行环境 二.项目搭建 三.测试 API 四.基础知识 五.写在最后 写在前面 C#语言可以创建RESTful服务,被称作WebApi.在这里总结学习使用支持创建.读取.更新.删除 ...

  4. ASP.NET Core 2.2 WebApi 系列【一】搭建ASP.NET Core WebApi项目

    一.步骤 从“文件”菜单中选择“新建”>“项目” . 选择“ASP.NET Core Web 应用程序”模板,再单击“下一步” . 将项目命名为 NetCoreWebApi,然后单击“创建” . ...

  5. ASP.NET Core 2.2 WebApi 系列【九】使用SignalR

    1.添加 SignalR 客户端库 右键点击项目->然后选择“添加” >“客户端库” 提供程序选择:unpkg ,库选择:@aspnet/signalr@1.1.4 选择“选择特定文件” ...

  6. ASP.NET Core 2.2 WebApi 系列【八】统一返回格式(返回值、模型验证、异常)

    现阶段,基本上都是前后端分离项目,这样一来,就需要前后端配合,没有统一返回格式,那么对接起来会很麻烦,浪费时间.我们需要把所有接口及异常错误信息都返回一定的Json格式,有利于前端处理,从而提高了工作 ...

  7. ASP.NET Core 2.2 WebApi 系列【七】泛型仓储模式和工作单元

    在之前的泛型仓储模式实现中,每个增删改都调用了SaveChanges方法,导致每次更新都提交了事务. 在实际开发过程中,我们经常遇到同时操作多张表数据,那么按照之前的写法,对数据库提交了多次操作,开启 ...

  8. ASP.NET Core 2.2 WebApi 系列【五】MiniProfiler与Swagger集成

    MiniProfiler 是一款性能分析的轻量级程序,可以基于action(request)记录每个阶段的耗时时长,还是可以显示访问数据库时的SQL(支持EF.EF Code First)等 一.安装 ...

  9. ASP.NET Core 2.2 WebApi 系列【四】集成Swagger

    Swagger 是一款自动生成在线接口文档+功能测试功能软件 一.安装程序包 通过管理 NuGet 程序包安装,搜索Swashbuckle.AspNetCore 二.配置 Swagger 将 Swag ...

随机推荐

  1. c++-面向对象类的示例-求周长面积,判断体积相等-文件操作和一般操作

    面向对象编程示例:求周长和面积 #define _CRT_SECURE_NO_WARNINGS #include <iostream> using namespace std; //圆的周 ...

  2. 松软科技Web课堂:JavaScript JSON

    JSON 是存储和传输数据的格式. JSON 经常在数据从服务器发送到网页时使用. 什么是 JSON? JSON 指的是 JavaScript Object Notation JSON 是轻量级的数据 ...

  3. injected stylesheet 谷歌扩展插件,造成样式异常

    今天在开发的时候,遇到一个问题,就是我们我在写发送广告的功能,然后我用了一个textare文本框,这个时候,发现了一个问题.这个文本框凭空消失了.不见了,我以为是自己的那个样式不小心把这个隐藏掉了后来 ...

  4. SAP B1:水晶报表中用Code128制作条型码的方法

    一.在[文件系统]中新建一个[Fonts文件夹],然后添加[Code128.ttf]文件. 二.在水晶报表里的[字段资源管理器]的[公式字段]中新建一个公式字段.点击[使用编辑器]之后弹出[公式工作室 ...

  5. 浅析椭圆曲线加密算法(ECC)

    本文首发于先知社区,原文链接:https://xz.aliyun.com/t/6295 数学基础 黎曼几何中的"平行线" 欧几里得<几何原本>中提出五条公设: 过相异两 ...

  6. Linux7 64安装 oracle 11g Error in invoking target 'agent nmhs' of makefile

    在makefile中添加链接libnnz11库的参数修改$ORACLE_HOME/sysman/lib/ins_emagent.mk,将$(MK_EMAGENT_NMECTL)修改为:$(MK_EMA ...

  7. Zuul 详解,带视频

    疯狂创客圈 Java 高并发[ 亿级流量聊天室实战]实战系列 [博客园总入口 ] 架构师成长+面试必备之 高并发基础书籍 [Netty Zookeeper Redis 高并发实战 ] 前言 Crazy ...

  8. CentOS7中_带sqlite3_CGO的golang程序_交叉编译到arm中

    CentOS7中_带sqlite3_CGO的golang程序_交叉编译到arm中 转载注明来源: 本文链接 来自osnosn的博客,写于 2019-10-28. 编写了个golang程序,用到了这个C ...

  9. [Codeforces 1244C] The Football Season

    思维加枚举 题意 :足球赛,赢平所得到的分数分别为w和d,w>d,分别求赢平输的场数,输出一组即可,即x+y+z=n 且 xw+yd=p的一组解. 可以扩展公约数做,但由于注意到d和w<1 ...

  10. CentOS 线上搭建 jupyter_server 笔记

    一.背景 为公司负责 Data Science 的同事配置线上 jupyter_server (jupyter + jupyter_kernel_gateway)环境. 二.环境 CentOS 7.6 ...