• ABP: 关于IRepository不继承IQueryable<TEntity>的问题 / lQueryable和IEnumerable的区别


    问题描述:

    之前版本中,我们可以看到IRepository继承自IQueryable接口如下图:

    我们可以通过Repository的AsQueryable()方法获取IQueryable对象或者使用Linq中为IQueryable提供的Where当方法。

    升级c# 6.0之后,仓储不再继承IQueryable,不能直接使用LINQ语句了。

    仓储中提供了方法GetQueryableAsync用于获取IQueryable对象用于代替原AsQueryable方式,用法如下:

    var query = await _repository.GetQueryableAsync();

    获取IQueryable对象后,我们依旧可以使用Linq提供的方法对其进行拼接和操作: 

    var query = (await _epository.GetQueryableAsync()).Where(x => x......

    ---------------------------------------------------fuck line----------------------------------------------------------------- 

    By the way 

    首先明确两个概念

    • IQueryable:提供对未指定数据类型的特定数据源的查询进行计算的功能。
    • IEnumerable:公开枚举数,该枚举数支持在非泛型集合上进行简单迭代。

    IQueryable 继承自 IEnumerable,它们俩最大的区别是,IQueryable 是表达式树处理,可以延迟查询,而 IEnumerable 只能查询在本地内存中,Repository 的概念就不多说了,在“伪 DDD”设计中,你可以把它看作是数据访问层。

    EF Core :IQueryable的延迟执行_董厂长的博客-CSDN博客

    下面我们先实现 Repository 返回 IEnumerable:

    1. public interface IBookRepository
    2. {
    3. Book GetById();
    4. IEnumerable GetAllBooks();
    5. IEnumerable GetBy....();
    6. void Add(Book book);
    7. void Delete(Book book);
    8. void SaveChanges();
    9. }

    上面是我们的一般接口设计,包含查询、增加、删除操作,你发现并没有修改,其实我们可以先通过 GetById 操作,然后取得 Book 对象,进行修改,最后执行 SaveChanges 就可以了,在持久化数据库的时候,会判断实体状态值的概念,最后进行应用改变

    GetBy....() 代表了一类查询方法,因为我们的业务比较复杂,对 Book 的查询会千奇百怪,所以,没有办法,我们只能增加各类查询方法来满足需求,最后可能导致的结果是,一个 Where 对应一个查询方法,IBookRepository 会充斥着各类查询方法,并且这些查询方法一般只会被一个 Application 方法调用,如果你查看下 GetBy....() 方法实现,会发现其实都大同小异,不同的只是 Where 条件,这样的结果就会导致代码变得非常的臃肿。

    针对上面的问题,怎么办呢?因为 IEnumerable 是查询在本地内存中,所以没有办法,我们只能这样处理,那如何使用 IQueryable 会是怎样的呢?我们看下代码:

    1. public interface IBookRepository
    2. {
    3. IQueryable GetBooks();
    4. void Add(Book book);
    5. void Delete(Book book);
    6. void SaveChanges();
    7. }

    只有一个 GetBooks 查询,那它能满足各类查询需求吗?我们看下 Application 中调用的代码:

    1. public class BookApplication : IBookApplication
    2. {
    3. private IBookRepository _bookRepository;
    4. public BookApplication(IBookRepository bookRepository)
    5. {
    6. _bookRepository = bookRepository;
    7. }
    8. public IEnumerable GetAllBooks()
    9. {
    10. return _bookRepository.GetBooks().AsEnumerable();
    11. }
    12. public IEnumerable GetBooksByUser(int userId)
    13. {
    14. return _bookRepository.GetBooks().Where(b => b.UserId == userId).AsEnumerable();
    15. }
    16. //....
    17. }

    因为 IQueryable 是延迟查询,只有在执行 AsEnumerable 的时候,才会真正去查询,也可以这么说,BookApplication 可以根据需求任意构建查询表达式树,理解为,再真正执行的时候才会生成一段完整的SQL‘语句。

    从上面的代码中,我们可以看到,IQueryable 很好的解决了使用 IEnumerable 所出现的问题,一个查询可以应对千变万化的应用查询,IQueryable 看起来好像是那么的强大,其实 IQueryable 的强大并不限于此,上面说的是查询表达式,那添加、修改和删除操作,可以使用它进行完成吗?修改和删除是可以的,添加并不能,具体可以参考 dudu 的这篇博文:开发笔记:基于EntityFramework.Extended用EF实现指定字段的更新

    关于 EntityFramework.Extended 的扩展,需要记录下,因为这个东西确实非常好,改变了我们之前的很多写法和问题,比如,在之前使用 EF 进行修改和删除实体,我们一般会这些写:

    1. public class BookApplication : IBookApplication
    2. {
    3. private IBookRepository _bookRepository;
    4. public BookApplication(IBookRepository bookRepository)
    5. {
    6. _bookRepository = bookRepository;
    7. }
    8. public void UpdateNameById(int bookId, string bookName)
    9. {
    10. var book = _bookRepository.GetById(bookId);
    11. book.BookName = bookName;
    12. _bookRepository.SaveChanges();
    13. }
    14. public void UpdateNameByIds(int[] bookIds, string bookName)
    15. {
    16. var books = _bookRepository.GetBooksByIds(bookIds);
    17. foreach (var book in books)
    18. {
    19. book.BookName = bookName;
    20. }
    21. _bookRepository.SaveChanges();
    22. }
    23. public void Delete(int id)
    24. {
    25. var book = _bookRepository.GetById(id);
    26. _bookRepository.Delete(book);//context.Books.Remove(book);
    27. _bookRepository.SaveChanges();
    28. }
    29. }

    上面的写法有什么问题呢?其实最大的问题就是,我们要进行修改和删除,必须先获取这个实体,也就是先查询再进行修改和删除,这个就有点多余了,尤其是 UpdateNameByIds 中的批量修改,先获取 Book 对象列表,然后再遍历修改,最后保存。仔细想想,还不如不用 EF 来的简单,因为一个 Update SQL 就可以搞定,简单并且性能又高,为什么还要使用 EF 呢?这是一个坑?其实使用 EF 也可以执行 SQL,但这就像换了个马甲,没有什么卵用。

    针对上面的问题,该如何解决呢?使用 EntityFramework.Extended 和 IQueryable 就可以,我们改造下上面的代码:

    1. using EntityFramework.Extensions;
    2. public class BookApplication : IBookApplication
    3. {
    4. private IBookRepository _bookRepository;
    5. public BookApplication(IBookRepository bookRepository)
    6. {
    7. _bookRepository = bookRepository;
    8. }
    9. public void UpdateNameById(int bookId, string bookName)
    10. {
    11. IQueryable books = _bookRepository.GetBooks();
    12. books = books.Where(b => b.bookId == bookId);
    13. books.Update(b => new Book { BookName = bookName });
    14. }
    15. public void UpdateNameByIds(int[] bookIds, string bookName)
    16. {
    17. IQueryable books = _bookRepository.GetBooks();
    18. books = books.Where(b => bookIds.Contains(bookIds));
    19. books.Update(b => new Book { BookName = bookName });
    20. }
    21. public void Delete(int id)
    22. {
    23. IQueryable books = _bookRepository.GetBooks();
    24. books = books.Where(b => b.bookId == id);
    25. books.Delete();
    26. }
    27. }

    有没有发现什么不同呢?原来 IQueryable 还可以这样写?不只是用于查询,也可以用于删除和修改,另外,通过追踪生成的 SQL 代码,你会发现,没有了 SELECT,和我们直接写 SQL 是一样的效果,在执行修改和删除之前,我们需要对查询表达树进行过滤,也就是说的,当我们最后应用修改的时候,会是在这个过滤的查询表达树基础上的,比如上面的 Delete 操作,我们先通过 bookId 进行过滤,然后直接进行 Delete 就可以了。

    当 BookApplication 操作变的简单的时候,BookRepository 也会相应变的简单:

    1. public interface IBookRepository
    2. {
    3. IQueryable GetBooks();
    4. void SaveChanges();//只用于Books.Add(book);
    5. }

    一个 IQueryable 表达树,一个 SaveChanges 操作,就可以满足 BookApplication 中的所有操作。

    关于 Repository 返回 IQueryable 的讨论,我大致总结下:

    好处:

    1. 延迟执行。
    2. 减少 Repository 重复代码(GetBy...)。
    3. IQueryable 提供更好的灵活性。
    4. ...

    坏处:

    1. 隔离单元测试。
    2. 数据访问在 Repository 之外完成。
    3. 数据访问异常在 Repository 之外抛出。
    4. 该领域层将充斥着这些相当详细查询。
    5. ...

    好处就不多说了,因为我们上面已经实践过了,关于坏处,“隔离单元测试”是什么意思呢?也就是说我们不能很好的对 Repository 进行单元测试,一方面是因为 IRepository 是那么的简单(就两个方法),另一方面 IQueryable 是查询表达树,它并不是完成时,只有在具体调用的时候才会查询完成,所以,对于 Repository 的单元测试,显然是没有任何意义的。

    关于 Repository Pattern and IQueryable 这篇博文,我想再说一下,因为这个老外的观点非常赞,首先,它是基于 Repository 模式概念基础上说的,所以,我们一开始说:在“伪 DDD”设计中,你可以把 Repository 看作是数据访问层。这是两个不同的前提,我再大致总结下这个老外的观点:

    • However the mistake is not the IQueryable itself, but its purpose.(不是 IQueryable 本身的错误,而是它的目的。)
    • The point is that using IQueryable, you're asking for a query builder and not for a model.(问题的关键是,使用 IQueryable 是一个查询生成器,而不是一个模型。)
    • we want to specify what to get, not how to get it.(我们想通过规约得到它,而不是怎样去得到。)
    • tell it the what, not the how.

    看了上面,是不是有点豁然开朗的感觉呢,其实从 Repository 的模式概念方面考虑,使用 IQueryable 确实不是很恰当,但不可否认的是,IQueryable 又这么强大和便利,怎么办呢?就像博文一开始强调的那样:Repository 的概念就不多说了,在“伪 DDD”设计中,你可以把它看作是数据访问层。

    所以呢,如果你的项目是“伪 DDD”,并且 Repository 是被你看作“数据访问层”,那么使用 IQueryable 就没啥问题了。

    Reference:

    Repository 返回 IQueryable?还是 IEnumerable?(转) - 为之守望 - 博客园

    ABP vNext5.0仓储获取IQueryable问题 - 车神ne - 博客园

  • 相关阅读:
    python设置全局代理
    Spring Boot : ORM 框架 JPA 与连接池 Hikari
    微型神经网络库MicroGrad-基于标量自动微分的类pytorch接口的深度学习框架
    用于多步乘客需求预测的深度时空序列建模
    MySql基础篇学习笔记
    GPT家族
    一种基于E3处理器平台的NAS完整方案(从电脑组装到网站部署)
    Radis基础命令(String类型)Value为JSON
    Spring MVC异常处理
    Gateway--服务网关
  • 原文地址:https://blog.csdn.net/dongnihao/article/details/126389101