【Yom框架】漫谈个人框架的设计之一:是IRepository还是IRepository<T>?
前言
对于仓储Repository的设计,其实很多人都很纠结,因为从广义来说,Repository有两种类型:
IRepository和IRepository<T>
框架的重构想得最多的最重要的几个问题:
1:解耦(每层可以替换其他的,比如换一个UI层可以把Web 项目快速转换成Winform项目)
2:扩展性(可以灵活抹去框架的某个层,让其他的第三方框架依据自己的接口实现该层的逻辑,其它层不变,也就是插拔式扩展)
3:灵活(开发便捷,使用灵活)
4:维护性(别人了解框架后,可以让别人无障碍维护)
........
-------------------------------------
题外话不多说 马上进入辩证主题:是IRepository还是IRepository<T> ?
------------------------------------
首先看IRepository<T>
IRepository<T>接口定义形式如下(其中IEntity是一个实体类接口):
public interface IRepository<T> where T : Entity.IEntity
{
T FindBy(string primaryKey);
IEnumerable<T> FindAll();
IEnumerable<T> FindAll(string where);
IEnumerable<T> FindAll(string where, string order);
IEnumerable<T> FindAll(int pageIndex, int pageSize, string where, string order, out int count);
void Add(T entity);
void Delete(T entity);
void DeleteAll();
void DeleteAll(string where);
void DeleteAll(System.Collections.IEnumerable pkValues);
void Update(T entity);
bool Exists(string primaryKey);
}
可以看见,IRepository和接口IEntity通过泛型T结合在了一起,形成了耦合
IRepository<T> 可以通过T操作IEntity
开发的时候,每个IEntity的子类都得对应一个IRepository<T>的子类,如:
public class DepartmentRepository : Repository.RepositoryBase<Entity.Department.Department>
{
}
其中Department是IEntity的一个子类
而RepositoryBase<T>是一个真正可用的仓储父类(此类已通过第三方或者自己的ORM框架实现了数据库操作)
------------------------------------
再看IRepository接口
------------------------------------
IRepository接口的设计:
public interface IRepository
{
#region 实体相关接口
TEntity FindBy<TEntity>(IEnumerable<string> primaryKey)
where TEntity : IEntity; IEnumerable<TEntity> FindAll<TEntity>() where TEntity : IEntity; IEnumerable<TEntity> FindAll<TEntity>(string where, params System.Data.IDataParameter[] ps) where TEntity : IEntity; IEnumerable<TEntity> FindAll<TEntity>(string where, string order, params System.Data.IDataParameter[] ps) where TEntity : IEntity; IEnumerable<TEntity> FindAll<TEntity>(int pageIndex, int pageSize, string where, string order, out int count, params System.Data.IDataParameter[] ps) where TEntity : IEntity; void Add<TEntity>(TEntity entity) where TEntity : IEntity; void Delete<TEntity>(TEntity entity) where TEntity : IEntity; void DeleteAll<TEntity>() where TEntity : IEntity; void DeleteAll<TEntity>(string where, params System.Data.IDataParameter[] ps) where TEntity : IEntity; void DeleteAll<TEntity>(IEnumerable<IEnumerable<string>> pkValues)
where TEntity : IEntity; void Update<TEntity>(TEntity entity) where TEntity : IEntity; bool Exists<TEntity>(IEnumerable<string> primaryKey)
where TEntity : IEntity; #endregion
#region 原始数据操作接口 int ExecuteSql(string sql, params System.Data.IDataParameter[] ps); object ExecuteScalar(string sql, params System.Data.IDataParameter[] ps); System.Data.DataTable ExecuteDataTable(string sql, params System.Data.IDataParameter[] ps);
#endregion
}
这种接口的设计就是把IReopository<T>里的T放入接口的方法中,
让泛型方法操作对应的带入泛型实体类
IReopository接口的设计可以很好地实现Repository共用
也就是说整个项目只要一个通过ORM实现了的RepositoryBase类就可以操作所有的持久层实体对象
不用每个实体类都对应一个Repository
大大的减少了项目开发的繁杂性
对于业务逻辑,新增一个Server层让每个Server类对应一个实体类的逻辑
如:假设有Class Aa 则必须有 Class AaServer对应
而Server就调用RepositoryBase类操作Server类对应的实体
--------------------------------------
总结:
其实不管是IRepository还是IRepository<T> 都各自有各自的优势:
1、IRepository<T>的子类对实体类是很专注的,它只可以操作一个实体类,对IRepository<T>的子类的修改不会影响到其他实体类的操作
从而可以实现对应实体类的个性化拓展;
2、IRepository可以操作所有的实体类,修改IRepository的子类则会影响所有的实体的操作
虽然如此,在开发过程中,难免会有在某个业务层XxServer操作其他实体类的需要
如果是用IRepository<T>仓储,那么必须在业务层XxServer中New很多其他实体类对应的IRepository<T>的子类对象出来
这对于Repository与Server的解耦是个大忌,也就是说Repository层和Server层已经高度耦合了。
也正因为这个原因我个人更倾向于IRepository,并抛弃Repository层(如果是每个实体对应一个Repository,那么将需要一个Repository层),
只让一个可以操作所有所有实体的Repository存在就可以了
更重要的原因是Repository层相对来说,接口比较稳定,一般的项目,没有要扩展IRepository接口操作的需要。
所以IRepository接口一个重要的优势是:
在某个实体类的Server层可以统一用IRepository类的方法实现业务,
不需要像IRepository<T>实现方式一样,New额外的对象就才可以操作其他实体类,
只要在【Repository.方法<T>】里的T换成其他实体类就可以了
这对解耦来说是有好处的。
所以正是因为这个原因,我选择IRepository,而不是IRepository<T>
---------------------
题外话
上面的IRepository接口已经被我再次抛弃了
抛弃原因如下:
1:接口的组合主键扩展性差,也就是说主键会受制于ORM框架的实现
2:不支持搜索和排序解耦
至于新的IRepository接口 将在下篇文章给出
-------------------------------------
完
【Yom框架】漫谈个人框架的设计之一:是IRepository还是IRepository<T>?的更多相关文章
- 【Yom框架】漫谈个人框架的设计之三:业务接口+UI层的设计(基于Castle实现的Repository)
Repository层设计的文章见:[http://www.cnblogs.com/yomho/p/3297042.html] 一.概要设计 上面Reposity 应该为 Repository 特 ...
- 【Yom框架】漫谈个人框架的设计之二:新的IRepository接口+搜索和排序解耦(+基于Castle实现)
经过了上篇IRepository和IRepository<T>的讨论[文章地址为:http://www.cnblogs.com/yomho/p/3296759.html] 我选择了IRep ...
- 框架的设计之IRepository还是IRepository<T>
[Yom框架]漫谈个人框架的设计之[是IRepository还是IRepository<T>]? 前言 ...
- 基于SSH框架的考勤管理系统的设计与实现
基于SSH框架的考勤管理系统的设计与实现
- 好的框架需要好的 API 设计 —— API 设计的六个原则
说到框架设计,打心底都会觉得很大很宽泛,而 API 设计是框架设计中的重要组成部分.相比于有很多大佬都认可的面向对象的六大原则.23 种常见的设计模式来说,API 设计确实缺少行业公认的原则或者说设计 ...
- 基于layui的框架模版,采用模块化设计,接口分离,组件化思想
代码地址如下:http://www.demodashi.com/demo/13362.html 1. 准备工作 编辑器vscode,需要安装liveServer插件在前端开启静态服务器 或者使用hbu ...
- js框架漫谈
现在实际项目中可供选择的javascript框架很多,热门的有jquery,dojo,mootools,ext等.这些框架按照不同的标准有不同的分类方法,比如按照扩展方式便可分为prototype式的 ...
- Android酷炫实用的开源框架(UI框架)
Android酷炫实用的开源框架(UI框架) 前言 忙碌的工作终于可以停息一段时间了,最近突然有一个想法,就是自己写一个app,所以找了一些合适开源控件,这样更加省时,再此分享给大家,希望能对大家有帮 ...
- Android酷炫实用的开源框架(UI框架) 转
Android酷炫实用的开源框架(UI框架) 前言 忙碌的工作终于可以停息一段时间了,最近突然有一个想法,就是自己写一个app,所以找了一些合适开源控件,这样更加省时,再此分享给大家,希望能对大家有帮 ...
随机推荐
- Spring学习使用标签来标记资源(@Component、@Repository、 @Service和@Controller)和用法(包括如何jsp正在使用)
首先,在xml其中新增部分标有下划线的文件,容器初始化的时候需要扫描包 注意: a. 包款扫描(下划线部分)一定要加,默认是不扫描整个包.与每一包之间','开.如过具有同样的父包,那么我们能够 ...
- css布局之选择切换按钮
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- CSharp设计模式读书笔记(18):中介者模式(学习难度:★★★☆☆,使用频率:★★☆☆☆)
中介者模式(Mediator Pattern):用一个中介对象(中介者)来封装一系列的对象交互,中介者使各对象不需要显式地相互引用,从而使其耦合松散,而且可以独立地改变它们之间的交互,中介者模式又称为 ...
- 【Java编码准则】の #12不要使用不安全或者强度弱的加密算法
安全性要求高的应用程序必须避免使用不安全的或者强度弱的加密算法,现代计算机的计算能力使得攻击者通过暴力破解能够攻破强度弱的算法.比如,数据加密标准算法DES是极度不安全的,使用类似EFF(Electr ...
- 【百度地图API】——如何让标注自动呈现在最佳视野内
原文:[百度地图API]--如何让标注自动呈现在最佳视野内 摘要: “我有一堆标注,不规则的散落在地图的各个地方,我想把它们展示在一个最佳视野中,怎么办呢?”一位API爱好者咨询道. -------- ...
- Emojicon - 为你提供emoji表情的整套方案
Github : https://github.com/rockerhieu/emojicon 事实上国内的<表情大全>也有提供类似整套表情解决方式,并且还支持gif表情:http://w ...
- SQL语句分享[不定期更新]
查询临时表 if object_id('')>0 查询表中的数据 select 'insert into ta1(col1,col2,col3) values('''+ltrim(列1)+''' ...
- 注意,WebDeploy服务会占用80端口。(Windows关闭了IIS,80端口任然被占用)
最近遇到一个很奇怪的事情,Windows上的 IIS 网站 全关掉了,80端口仍然被占用.然后我新装了一台服务器,一个一个组件地装,装一个测一次,最后发现,WebDeploy这个组件,会占用80端口. ...
- ubuntu下使用openocd+jlink进行STM32开发调试
安装openocd就不用多说了,使用 apt-get install openocd 这个命令就可以做到. 对于使用stm32w系列的MCU,需要下载新的openocd-0.7及以上版本才能支持.0. ...
- 远程控制编写之屏幕传输 MFC实现 屏幕截图 发送bmp数据 显示bmp图像
远程控制编写之屏幕传输 MFC实现 屏幕截图 发送bmp数据 显示bmp图像: 一 : 首先要了解bmp图像的结构 详情请看我转载的一篇文章http://blog.csdn.net/hnust_x ...