【Yom框架】漫谈个人框架的设计之【是IRepository还是IRepository<T>】?

前言                                                                                                                 

对于仓储Repository的设计,其实很多人都很纠结,因为从广义来说,Repository有两种类型:

IRepositoryIRepository<T>

框架的重构想得最多的最重要的几个问题:

1:解耦(每层可以替换其他的,比如换一个UI层可以把Web 项目快速转换成Winform项目)

2:扩展性(可以灵活抹去框架的某个层,让其他的第三方框架依据自己的接口实现该层的逻辑,其它层不变,也就是插拔式扩展)

3:灵活(开发便捷,使用灵活)

4:维护性(别人了解框架后,可以让别人无障碍维护)

........

-------------------------------------

题外话不多说 马上进入辩证主题:是IRepository还是IRepository<T> ?

------------------------------------

首先看IRepository<T>                                                                   

IRepository<T>接口定义形式如下(其中IEntiry是一个实体类接口):

 1 public interface IRepository<T> where T : Entity.IEntity
2 {
3 T FindBy(string primaryKey);
4 IEnumerable<T> FindAll();
5 IEnumerable<T> FindAll(string where);
6 IEnumerable<T> FindAll(string where, string order);
7 IEnumerable<T> FindAll(int pageIndex, int pageSize, string where, string order, out int count);
8 void Add(T entity);
9 void Delete(T entity);
10 void DeleteAll();
11 void DeleteAll(string where);
12 void DeleteAll(System.Collections.IEnumerable pkValues);
13 void Update(T entity);
14 bool Exists(string primaryKey);
15 }

可以看见,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
}

这种接口的设计就是把ReopositoryBase<T>里的T放入接口的方法中,

让泛型方法操作对应的实体类

这种接口的设计可以很好地实现Repository共用

也就是说整个项目只要一个通过ORM实现了的RepositoryBase类就可以操作所有的持久层实体对象

不用每个实体类都对应一个IRepository

大大的减少了项目开发的便捷性

对于业务逻辑,新增一个Server层

Server层,让每个Server类对应一个实体类的逻辑,如:假设有Class Aa 则必须有 Class AaServer对应

而Server就调用RepositoryBase类操作Server类对应的实体

--------------------------------------

总结:                                                                                                                     

其实不管是IRepository还是IRepository<T> 都各自有各自的优势

IRepository<T>的子类对实体类是很专注的,它只可以操作一个实体类,对它的修改不会影响到其他实体来的操作

从而可以实现对应实体类的个性化拓展,而IRepository可以操作所有的实体类,修改了子类则会影响所有的实体的操作

虽然如此,在开发过程中,难免会有在某个业务层操作其他对象的需要

如果是IRepository<T>,那么必须在业务层New很多其他实体类对应的IRepository<T>子类对象出来

这对于解耦是个大忌,也就是说Repository层和Server层已经高度耦合了。

也正因为这个原因我个人更倾向于IRepository,抛弃Repository层,只让一个可以操作所有所有实体的Repository存在就可以了

更重要的原因是Repository层相对拉来说,接口比较稳定,一般的项目,没有必要扩展IRepository接口的操作。

 所以IRepository接口一个重要的优势是:

在某个实体类的Server层可以用IRepository类的方法,不需要New额外的对象就可以操作其他实体类,

只要在【Repository.方法<T>】里的T换成其他实体类就可以了

如果换成IRepository<T>实现,则需要在该实体类的Server层New很多其他实体类对应的IRepository<T>才可以操作其他实体类的对象,这对解耦来说是大忌。所以正是因为这个原因,我宁愿选择IRepository,而不是IRepository<T>

---------------------

题外话

上面的IRepository接口已经被我再次抛弃了

抛弃原因如下:

1:接口的组合主键扩展性差,也就是说主键会受制于ORM框架的实现

2:不支持搜索和排序解耦

所以建议自己好好设计IRepository接口

-------------------------------------

 
 

框架的设计之IRepository还是IRepository<T>的更多相关文章

  1. 【Yom框架】漫谈个人框架的设计之一:是IRepository还是IRepository<T>?

    前言                                                                                                   ...

  2. 【Yom框架】漫谈个人框架的设计之二:新的IRepository接口+搜索和排序解耦(+基于Castle实现)

    经过了上篇IRepository和IRepository<T>的讨论[文章地址为:http://www.cnblogs.com/yomho/p/3296759.html] 我选择了IRep ...

  3. 【Yom框架】漫谈个人框架的设计之三:业务接口+UI层的设计(基于Castle实现的Repository)

    Repository层设计的文章见:[http://www.cnblogs.com/yomho/p/3297042.html]   一.概要设计 上面Reposity 应该为 Repository 特 ...

  4. [连载]《C#通讯(串口和网络)框架的设计与实现》- 11.调试器的设计

    目       录 第十一章     调试器设计... 2 11.1         调试接口... 2 11.2         界面方式调试... 3 11.3         命令行方式调试.. ...

  5. [连载]《C#通讯(串口和网络)框架的设计与实现》-1.通讯框架介绍

    [连载]<C#通讯(串口和网络)框架的设计与实现>- 0.前言 目       录 第一章           通讯框架介绍... 2 1.1           通讯的本质... 2 1 ...

  6. [连载]《C#通讯(串口和网络)框架的设计与实现》-2.框架的总体设计

    目       录 C#通讯(串口和网络)框架的设计与实现... 1 (SuperIO)- 框架的总体设计... 1 第二章           框架总体的设计... 2 2.1           ...

  7. Mybatis原理分析之二:框架整体设计

    1.引言 本文主要讲解Mybatis的整体程序设计,理清楚框架的主要脉络.后面文章我们再详细讲解各个组件. 2.整体设计 2.1 总体流程 (1)加载配置并初始化       触发条件:加载配置文件 ...

  8. Java日志系统框架的设计与实现

    推荐一篇好的文章介绍java日志系统框架的设计的文章:http://soft.chinabyte.com/database/438/11321938.shtml 文章内容总结: 日志系统对跟踪调试.程 ...

  9. activiti5.13 框架 数据库设计说明书

    转载自:http://www.cnblogs.com/llzgzljl/p/3356108.html activiti5.13 框架 数据库设计说明书 1.结构设计 1.1.    逻辑结构设计 Ac ...

随机推荐

  1. MAC使用小技巧(二)

    一.Safari-->广告数量不足 --原因:DNS被拦截,被恶意推送广告. ----------------------------- [ 思路 ] 修改hosts文件 $ cd /etc $ ...

  2. C# Parse和Convert的区别分析

    原文:C# Parse和Convert的区别分析 大家都知道在进行类型转换的时候有连个方法供我们使用就是Convert.to和*.Parse,但是疑问就是什么时候用C 什么时候用P 通俗的解释大家都知 ...

  3. SQL代理执行EXE可执行程序

    原文:SQL代理执行EXE可执行程序  1.如果没有启用xp_cmdshell安全配置是不可以使用的-- 启用xp_cmdshellEXEC sp_configure 'xp_cmdshell', 1 ...

  4. JS判断Array数组中是否包含指定元素

    1.调用方式: var arr=["a","b"]; alert(arr.in_array("a")) 2.JS判断数组是否包含指定元素方法 ...

  5. 收集的VS2013的使用小技巧( 不断总结中)

    对于经常使用vs的朋友,如果能用键盘直接做的事,还是键盘更便捷点,现在我就把自己遇到的一些给写下来. 1.对一个函数的说明 先写一个函数,以及参数,完成后,在函数上输入///,vs会自动补全说明的信息 ...

  6. unity3d插件Daikon Forge GUI 中文教程-1-Daikon Forge介绍

    (游戏蛮牛首发)大家好我是孙广东官网提供了专业的视频教程http://www.daikonforge.com/dfgui/tutorials/,只是是在youtube上,要观看是须要FQ的. 只是教程 ...

  7. Android开发之Mediaplayer

    Android提供了常见的音频.视频的编码.解码机制.借助于多媒体类MediaPlayer的支持,开发者能够非常方便在在应用中播放音频.视频.本篇博客主要解说在Android平台下怎样播放一个音频文件 ...

  8. 国内首篇介绍JanOS物联网操作系统的文章 - 如何把你的手机主板打造成物联网平台

    天地会珠海分舵注:如无意外,您现在正在看的将是国内首篇且是唯一一篇介绍炙手可热的物联网的操作系统JanOS的文章!不信你去百度!希望大家能喜欢.但本文只是引言,更多信息请还是访问JanOS的官网:ht ...

  9. Monkey源码分析之事件源

    上一篇文章<Monkey源码分析之运行流程>给出了monkey运行的整个流程,让我们有一个概貌,那么往后的文章我们会尝试进一步的阐述相关的一些知识点. 这里先把整个monkey类的结构图给 ...

  10. Installshield关于.NET安装时需要重启动的处理办法,以及延伸出的重启后继续安装的安装包的一点想法

    原文:Installshield关于.NET安装时需要重启动的处理办法,以及延伸出的重启后继续安装的安装包的一点想法 很多朋友做安装包的时候,所打包的软件需要.NET Framework之类的环境,他 ...