基础知识---IEnumerable、ICollection、IList、IQueryable
一、定义
IEnumerable
public interface IEnumerable<out T> : IEnumerable
ICollection
public interface ICollection<T> : IEnumerable<T>, IEnumerable
IList
public interface IList<T> : ICollection<T>, IEnumerable<T>, IEnumerable
List
public class List<T> : IList<T>, ICollection<T>, IEnumerable<T>, IList, ICollection, IEnumerable
可以看到功能上List最强大,性能上IEnumerable更好,其实性能上都差不多,都是接口。
二、IEnumerable<T>和IQueryable<T> 的区别
IEnumerable<T> 是linq to object。
IQueryable<T> 是linq to sql。
打个比方,我从Users中获取1条数据 var q=db.Users.orderby(x=>x.id).Take(1);
对于IEnumerable<T>来说,他先会把所有数据加载到内存,然后在取一条数据。
对于IQueryable<T> 来说,他会生成一个sql语句,只是取一条数据
三、接口设计中,函数的返回值应该怎么选择?
留意 .net 的类库,会发现很少有方法返回 List<T> 或 IList<T> 的,但是很多 .net 的程序员喜欢这么做,为什么?因为 List<T> 的功能实在太强大了吧。
IEnumerable<T> 是延时求值的,如果在调用栈里面输入和输出都是 IEnumerable<T> 的话,是最好的,因为无论调用多少次,循环都只有一次,.NET 里面的 lambda 就是这样的。
但是调用栈的输入输出无法确定的时候,情况就比较复杂。如果你返回的是 IEnumerable<T>,有可能调用栈的上一层执行了它的迭代器,造成并不需要的多次循环。
如果调用栈的上一层使用的是 List<T> 的 Copy() 方法,那么你返回 List<T> 会比较好,如果你返回的是 IEnumerable<T>,一般是会被 ToList() 再 Copy()
所以这个问题关键还是看怎么使用返回值,来决定你编写的函数的返回值。
使用尽可能底层的、具体的类型。不要从一开始就超过你的需求来搞“抽象”。因此首先是使用 List<T>。
能返回List就不要返回IList:如果后面有方法需要IList可以直接传;如果后面需要用到List的属性就能直接用
如果返回的是IList:后面有方法需要List,你得想想我能不能转,或者回去改;如果需要用到List的属性,还得回去改,综上考虑返回范围更广的类型好处多多。
但是,随着你真正开始需要抽象了,你就“被迫”需要重构为抽象的类型了。这时候再重构,要在必须的时候才进行抽象。真正会抽象的人,肯定是反而不滥用抽象,是在必须的时候才抽象的。这个道理大概是“从不抽象设计”的人总也不能理解的。
有的人,很实际,你看到他的代码就知道他的这部分程序设计已经到达那个程度、必须抽象才能用在系统中。其返回的具体数据类型“确实是”多种子类型定义的,所以才抽象。这就是把复杂的接口弄得刚刚合适的声明程度。
而有的人,代码的抽象总是高于实际,中间存在着含糊的、不确定的因素,明明其输入输出的数据类型不具有抽象、多态处理,却要弄一个很抽象的定义。这就是把简单的事情弄复杂了。
假设说你设计的时候,你有多种实际子类型的考虑,你确实有多种实现代码,那么你就返回一个枚举类型,明确地告诉使用者“我不能返回集合类型,我不保证实现所有的集合类型才有的功能”。这是因为你现在要实现的东西,确实是不能用集合类型的。
反过来说,如果你输出的东西只能用集合类型,那么就返回集合类型,方便调用者傻瓜化地使用。将来还有其它复杂的扩展,那么应该重构接口,增加重载新方法,而不用去修改原来使用集合类型的东西。比如说 Linq 有一系列的基于枚举类型的功能方法,但是不是说原来基于集合类型而定义的一大堆方法就不应该设计出来。不需要放这个马后炮。所以根据现在实际情况,定义接口,习惯于经常在新版本中重构接口,而不是纠结在“一次性‘最完美’不再变”的接口,就可以了。
比如多层项目中,数据层,业务层,展现层三层中,从数据层查询数据集到展现层需要“传递”2次,如果是List或者IEnumerable,会在数据层就将数据加载到内存,另外对于一个方法返回List集合,接受数据时每次都new 一个List集合 我感觉是不妥的,如果数据量较大是很损性能。 而IQueryable对于数据的操作,会根据自定义的扩展方法 来生成最终的查询语句,而不是从一开始就加载数据,这就达到了哪里需要数据就在哪里查询。其实很多项目中都会有直接返回LIst的方法或者接口,这些方式难免是那些从培训中心从来的或者从一些书籍上看到的开发人员编写的。
抽象都是恰到好处地抽象,才叫抽象!
基础知识---IEnumerable、ICollection、IList、IQueryable的更多相关文章
- 服务器文档下载zip格式 SQL Server SQL分页查询 C#过滤html标签 EF 延时加载与死锁 在JS方法中返回多个值的三种方法(转载) IEnumerable,ICollection,IList接口问题 不吹不擂,你想要的Python面试都在这里了【315+道题】 基于mvc三层架构和ajax技术实现最简单的文件上传 事件管理
服务器文档下载zip格式 刚好这次项目中遇到了这个东西,就来弄一下,挺简单的,但是前台调用的时候弄错了,浪费了大半天的时间,本人也是菜鸟一枚.开始吧.(MVC的) @using Rattan.Co ...
- 如何选择使用IEnumerable, ICollection, IList
IEnumerable, ICollection, IList,每种接口只适合某些特定场景,如何区别使用呢? IEnumerable接口,只提供了一个获取迭代器的方法,这也是为什么可以使用foreac ...
- IEnumerable<> ICollection <> IList<> 区别
IEnumerable< ICollection < IList区别 public interface IEnumerable { IEnumerator GetEnumerator(); ...
- 【转载】我也说 IEnumerable,ICollection,IList,List之间的区别
做C#的同学们,都知道,一类只能有一个继承类,但可以实现多个接口.这句话就告诉我们:IEnumerable,ICollection,IList,List区别了 首先我看看 IEnumerable: / ...
- Asp.Net IEnumerable,ICollection,IList,List区别
做C#的同学们,都知道,一类只能有一个继承类,但可以实现多个接口.这句话就告诉我们:IEnumerable,ICollection,IList,List区别了 首先我看看 IEnumerable: / ...
- IEnumerable,ICollection,IList,List区别
做C#的同学们,都知道,一类只能有一个继承类,但可以实现多个接口.这句话就告诉我们:IEnumerable,ICollection,IList,List区别了 首先我看看 IEnumerable: / ...
- IEnumerable,ICollection,IList,List之间的区别
做C#的同学们,都知道,一类只能有一个继承类,但可以实现多个接口.这句话就告诉我们:IEnumerable,ICollection,IList,List区别了 // 摘要: // 公开枚举器,该枚举器 ...
- IEnumerable,ICollection,IList,List的使用
做C#的都知道:一类只能有一个继承类,但可以实现多个接口.这句话就告诉我们:IEnumerable,ICollection,IList,List区别了 首先我看看 IEnumerable: // ...
- (原创)(C#随笔)IEnumerable< ICollection < IList区别
public interface IEnumerable { IEnumerator GetEnumerator(); } 再看ICollection<T> public interfac ...
随机推荐
- 高强度学习训练第四天总结:JVM+Redis
JVM 复习了JVM堆内存的几个模块. 复习了JVM的几个控制工具. 复习了JVM发展历史 Redis 复习了Redis的事务控制.
- vue-router简易的实现原理
class VueRouter { constructor(options) { this.$options = options; this.routeMap = {}; // 路由响应式 this. ...
- Mac OSX(Mac OS10.11) 安装 pwntools 失败的最新解决方案
pwntools是一个 CTF 框架和漏洞利用开发库,用 Python 开发,由 rapid 设计,旨在让使用者简单快速的编写 exploit. 网上针对 Mac OS 的安装教程大多都是基于 pip ...
- HeadFirst设计模式---装饰者
定义装饰者模式 装饰者模式动态地将责任附加到对象上,若要扩展功能,装饰者提供了比继承更有弹性的替代方案.这句话摘自书中,给人读得很生硬难懂.通俗地来说,装饰者和被装饰者有相同的父类,装饰者的行为组装着 ...
- spring boot的异常处理
原文:https://blog.csdn.net/tianyaleixiaowu/article/details/70145251 全局异常处理是个比较重要的功能,一般在项目里都会用到. 我大概把一次 ...
- 数据库系统(五)---MySQL基础
一.SQL基本概念: SQL 已经成为关系数据库的标准语言,是一种数据库查询和程序设计语言,用 于存取数据以及查询.更新和管理关系数据库系统. 功能不仅仅是查询,还包括数据定义.数据操纵和数据控制等于 ...
- Uniform Buffer
Uniform Buffer 是一个很有用的缓存,可以将大量的需要传递至多个着色器的矩阵.向量数据等存储在uniform buffer中.这是一个公共的缓存,所以当多个着色器需要传递相同的数据时,可以 ...
- mysql中的case when then 的用法
将立立饭饭的性别变更为女 ,烦烦 嗯嗯 问我的性别变更为男 update `table` set sex = ( case when sname in('立立','饭饭') then '女' else ...
- URL处理----拼接和编码
ps:浪了好几周,我的锅... 前几天想爬取一个用户网站自动创建每个用户的资料方便注册一些账号,想写一个通用点的爬虫程序爬取只要配置一些爬取规则.爬取深度就ok,避免代码改动,由于时间关系只完成的个半 ...
- 201871010123-吴丽丽 《面向对象程序设计(Java)》第十周学习总结
项目 内容 这个作业属于哪个课程 https://www.cnblogs.com/nwnu-daizh/ 这个作业要求在哪里 https://www.cnblogs.com/nwnu-daizh/p/ ...