统一的仓储接口
目前,各种数据访问组件多如牦牛,提供接口各不相同,虽然大多都支持不同数据存储,但是还是存在一些问题,
有的团队成员喜欢用EF,有的喜欢用NHibernate,口味难调。于是我希望能有一组标准的接口,这样可以统一
代码的,也可以满足各成员的喜好。
目前来说,linq的出现,大大简化了代码的冗余和复杂性,
所以这组接口是建立在linq的基础上的,支持linq的数据访问组件目前有EF,NHibernate,IQtoolkit,Dblinq等。
这里面有两个重要的对象,Context(上下文对象)和Repository(仓储对象)。
我希望调用者只需要尽量少的对象和尽量少的方法。使得接口清晰简单。
首先上下文对象Context,它能够获得仓储对象并完成事务提交即可。
{
string Name { get; set; }
IRepository GetRepository<T>();
bool SaveChanges();
}
接着仓储对象Repository,它则可以对T类型的数据进行CRUD,并且还能像IQueryable那样支持很多操作。
public interface IRepository<T> : IQueryable<T> where T : class,IEntity, new()
T CreateModel();
bool New(T entity);
bool Update(T entity);
bool Delete(T t);
bool VirtualDelete(T t);
Guid NewGuid();
decimal NewId();
}
仓储对不同数据访问组件有些共同的东西,加之实现IQueryable,所以增加抽象类AbstractRepository
{
public IObjectContext Context { get; private set; }
public AbstractRepository(IObjectContext context)
{
Context = context;
}
protected abstract IQueryable<T> GetQueryable();
public abstract bool New(T entity);
public abstract bool Update(T entity);
public abstract bool Delete(T t);
public abstract bool VirtualDelete(T t);
public virtual T CreateModel()
{
return new T();
}
public virtual Guid NewGuid() { return Guid.NewGuid(); }
public virtual decimal NewId() { return decimal.Zero; }
public IEnumerator<T> GetEnumerator()
{
return GetQueryable().GetEnumerator();
}
IEnumerator IEnumerable.GetEnumerator()
{
return GetQueryable().GetEnumerator();
}
public Type ElementType
{
get { return GetQueryable().ElementType; }
}
public Expression Expression
{
get { return GetQueryable().Expression; }
}
public IQueryProvider Provider
{
get { return GetQueryable().Provider; }
}
}
主要的就需要这三个接口
代码写起来就是这样子
using (var context = ContextFactory.Create(contextName))
var r = context.GetRepository<T>();
var instance = r.CreateModel();
instance.Copy(tdto);
instance.Validate(context);
r.New(instance);
context.SaveChanges();
return instance;
}
这些重复性的代码,直接扩展到接口上,封装起来,外面只需继承此接口。
下面是EF的仓储实现
public class EFRepository<T> : AbstractRepository<T> where T : class,IEntity, new()
ObjectContext _context = null;
public EFRepository(IObjectContext context) : base(context) {
_context = base.Context as ObjectContext;
if (_context == null) { throw new Exception("EFRepository中上下文对象不是System.Data.Objects.ObjectContext对象。"); }
}
public override bool New(T entity)
{
GetObjectSet().AddObject(entity);
return true;
}
public override bool Update(T entity)
{
return true;
}
public override bool Delete(T t)
{
GetObjectSet().DeleteObject(t);
return true;
}
public override bool VirtualDelete(T t)
{
return true;
}
protected override IQueryable<T> GetQueryable()
{
return GetObjectSet();
}
ObjectSet<T> GetObjectSet()
{
return _context.CreateObjectSet<T>();
}
}
其它的仓储实现也比较简单,附上源码
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 地球OL攻略 —— 某应届生求职总结
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 提示词工程——AI应用必不可少的技术
· .NET周刊【3月第1期 2025-03-02】