我正在考虑改进我当前的存储库GetAll
方法实现并在下面编写了代码。我试图在谷歌中找到这种方法的想法,但没有成功。因此,请查看代码并帮助我回答以下一些问题。这是简化的示例:
class Service
{
public void DoOperation()
{
// Let's say we need to retrieve users by criteria
var items = UnitOfWork
.Over<User>() // Get repository of User
.GetAll() // Start querying
// If we need simple WHERE then use:
.Where(x => x.Email == "some@email.com")
// If we need more complex condition then use specification:
.Using(new UserNameContains("John"))
// Execute:
.List();
}
}
class UnitOfWork
{
public Repository<T> Over<T>()
{
// Get from DI container or injected field
return new Repository<T>();
}
}
class Repository<T>
{
public QueryWrapper<T> GetAll()
{
return new QueryWrapper<T>(Session.QueryOver<T>());
}
}
class QueryWrapper<T>
{
// Query from DB session. Init from constructor.
private IQueryOver<T> _query;
public QueryWrapper<T> Where(Expression<Func<T, bool>> expression)
{
_query = _query.Where(expression);
return this;
}
public QueryWrapper<T> Using(Specification<T> spec)
{
var spec = new TSpec();
_query = spec.Apply(_query);
return this;
}
public IEnumerable<T> List()
{
return return _query.List();
}
}
abstract class Specification<T>
{
public abstract IQueryOver<T> Apply(IQueryOver<T> query);
}
class UserNameContains : Specification<User>
{
// Init from constructor:
private string _name;
public override IQueryOver<User> Apply(IQueryOver<User> query)
{
return /* apply filter condition here */;
}
}
因此,我们得到了这样的好处:
- 不再需要创建自定义回购。单个泛型就足够了。
- 自定义规范实现现在与数据层分离并且可测试。
- 易于使用的服务。
- 我们总是能够扩展
QueryWrapper
以支持其他方法,例如OrderBy
等。 - 它仍然是可单元测试的。
您能否指出我的方法的漏洞或提供您对问题的看法?现有文章的链接也很棒。