博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
分享基于Entity Framework的Repository模式设计(附源码)
阅读量:6331 次
发布时间:2019-06-22

本文共 6012 字,大约阅读时间需要 20 分钟。

关于Repository模式,在这篇文章中有介绍, 

这篇文章介绍的是使用Entity Framework实现的Repositoy模式设计,欢迎各位拍砖.

阅读目录:

一、实现的思路和结构图

二、Repository设计具体的实现代码

三、Repository设计的具体的使用

四、总结

一,实现的思路和结构图

总结一下,Repository在实际使用中,有下面三种特点:

Repository的共同性

有一些公共的方法(增删改查), 这些方法无关于Repository操作的是哪个实体类,可以把这些方法定义成接口IRepository<TEntity>, 然后有个基类BaseRepository<TEntity>实现该接口的方法。

常见的方法,比如Find, Filter, Delete, Create等

Repository的差异性

每个Repository类又会有一些差异性,应当允许它们能够继承BaseRepository<TEntity>之外,还能够再扩展自己的一些方法。所以每个类都可以再定义一个自己特有的接口,定义一些属于自己Repository的方法。

Repository的协同性

不同的Repository可能需要协同,Repository对数据的修改,需要在统一的保存.

最终实现的类结构图如下:

二,Repository设计具体的实现代码

IRepository<TEntity>接口定义了Repository共有的方法, BaseRepository<TEntity>实现了这些接口的方法。其它的Repository类再集成BaseRepository<TEntity>方法,就天然的得到了对数据操作的基本方法。

IRepository<TEntity>代码

public interface IRepository
where TEntity : class{ IQueryable
All(); IQueryable
Filter(Expression
> predicate); IQueryable
Filter(Expression
> filter, out int total, int index = 0, int size = 50); bool Contains(Expression
> predicate); TEntity Find(params object[] keys); TEntity Find(Expression
> predicate); TEntity Create(TEntity t); void Delete(TEntity t); void Delete(Expression
> predicate); void Update(TEntity t); TEntity Single(Expression
> expression);}

BaseRepository<TEntity>代码

public class BaseRepository
: IRepository
where TEntity : class { protected readonly DbContext Context; public BaseRepository(DbContext context) { Context = context; } public TEntity FirstOrDefault(Expression
> expression) { return All().FirstOrDefault(expression); } public IQueryable
All() { return Context.Set
().AsQueryable(); } public virtual IQueryable
Filter(Expression
> predicate) { return Context.Set
().Where
(predicate).AsQueryable
(); } public virtual IQueryable
Filter(Expression
> filter, out int total, int index = 0, int size = 50) { var skipCount = index * size; var resetSet = filter != null ? Context.Set
().Where
(filter).AsQueryable() : Context.Set
().AsQueryable(); resetSet = skipCount == 0 ? resetSet.Take(size) : resetSet.Skip(skipCount).Take(size); total = resetSet.Count(); return resetSet.AsQueryable(); } public virtual void Create(TEntity TObject) { Context.Set
().Add(TObject); } public virtual void Delete(TEntity TObject) { Context.Set
().Remove(TObject); } public virtual void Update(TEntity TObject) { try { var entry = Context.Entry(TObject); Context.Set
().Attach(TObject); entry.State = EntityState.Modified; } catch (OptimisticConcurrencyException ex) { throw ex; } } public virtual int Delete(Expression
> predicate) { var objects = Filter(predicate); foreach (var obj in objects) Context.Set
().Remove(obj); return Context.SaveChanges(); } public bool Contains(Expression
> predicate) { return Context.Set
().Any(predicate); } public virtual TEntity Find(params object[] keys) { return Context.Set
().Find(keys); } public virtual TEntity Find(Expression
> predicate) { return Context.Set
().FirstOrDefault
(predicate); } }

IUnitOfWork接口定义了方法获取特定的Repository, 执行存储过程, SaveChange方法提交修改,统一更新数据。

IUnitOfWork接口代码:

public interface IUnitOfWork : IDisposable{      TRepository GetRepository
() where TRepository : class; void ExecuteProcedure(string procedureCommand, params object[] sqlParams); void SaveChanges();}

 

UnitOfWork代码, 代码中使用到了Autofac中的IComponentContext来获取Repository实例

public class UnitOfWork : IUnitOfWork{       private readonly IComponentContext _componentContext;       protected readonly DbContext Context;       public UnitOfWorkRepository(DbContext context, IComponentContext componentContext)       {           Context = context;           _componentContext = componentContext;       }       public TRepository GetRepository
() where TRepository : class { return _componentContext.Resolve
(); } public void ExecuteProcedure(string procedureCommand, params object[] sqlParams) { Context.Database.ExecuteSqlCommand(procedureCommand, sqlParams); } public void SaveChanges() { Context.SaveChanges(); } public void Dispose() { if (Context != null) Context.Dispose(); }}

三, Repository设计的具体的使用

这里我们定义一个操作Student的Repository类,看看如何实际用于开发中。这里加入StudentRepository有自己特定的方法,需要获取所有的Students,这个扩展的方法名字叫GetAllStudents

那么定义一个接口IStudentRepository, 包含了方法GetAllStudents(), 同时继承IRepository<Student>接口

public interface IStudentRepository : IRepository
{ IEnumerable
GetAllStudents();}

然后定义StudentRepository类来实现这个接口

public class StudentRepository : BaseRepository
, IStudentRepository{ private readonly SchoolContext _context; public StudentRepository(SchoolContext context) : base(context) { _context = context; } public IEnumerable
GetAllStudents() { return _context.Students; }}

使用Repository的代码如下:

IUnitOfWork unitOfWork = new UnitOfWork();var studentRepository = unitOfWork.GetRepository
();var students = studentRepository.GetAllStudents();//同时你也可以使用定义于IRepository
中的方法, 比如//unitOfWork.Delete(students.First());//unitOfWork.SaveChanges();

四,总结

上面的设计,把Repository的通用代码剥离到父类中,同时又允许每个Repository扩展自己的方法,达到了比较理想的状态。

只是现在的设计和Autofac耦合了,但是想剥离Autofac的话,直接使用下面的方式获取IStudentRepository的实例就很困难。

unitOfWorkRepository.GetRepository<IStudentRepository>();

 

如果有什么好的办法,欢迎指教。也欢迎各位拍砖。

 

最后,附上本文的相关源代码. 有朋友反映这个设计有问题,希望大家批判的看待。如果有不同看法,欢迎指教。

五, 反馈及更新

感谢热心的园友提供的意见。 这个Repository的设计是不成熟的,在使用了一段EF和看了一些文章之后,自己的一些探索和思考,还没有应用到实际项目中。

,,  中提到的UnitOfWork问题,已经修改好.

 提到的Single方法不合适,已经改成FirstOrDefault()

最新的源代码 

 

转载于:https://www.cnblogs.com/webenh/p/7691746.html

你可能感兴趣的文章
log4j Test
查看>>
HDU 1255 覆盖的面积(矩形面积交)
查看>>
Combinations
查看>>
SQL数据库无法附加,提示 MDF" 已压缩,但未驻留在只读数据库或文件组中。必须将此文件解压缩。...
查看>>
第二十一章流 3用cin输入
查看>>
在workflow中,无法为实例 ID“...”传递接口类型“...”上的事件“...” 问题的解决方法。...
查看>>
获取SQL数据库中的数据库名、所有表名、所有字段名、列描述
查看>>
Orchard 视频资料
查看>>
简述:预处理、编译、汇编、链接
查看>>
调试网页PAIP HTML的调试与分析工具
查看>>
路径工程OpenCV依赖文件路径自动添加方法
查看>>
玩转SSRS第七篇---报表订阅
查看>>
WinCE API
查看>>
POJ 3280 Cheapest Palindrome(DP 回文变形)
查看>>
oracle修改内存使用和性能调节,SGA
查看>>
SQL语言基础
查看>>
对事件处理的错误使用
查看>>
最大熵模型(二)朗格朗日函数
查看>>
深入了解setInterval方法
查看>>
html img Src base64 图片显示
查看>>