天衣有缝 发表于 2024-6-20 08:12:07

C#.Net筑基-集合知识全解


01、集合基础知识

.Net 中提供了一系列的管理对象集合的类型,数组、可变列表、字典等。从类型安全上集合分为两类,泛型集合 和 非泛型集合,传统的非泛型集合存储为Object,需要类型转。而泛型集合提供了更好的性能、编译时类型安全,推荐使用。
.Net中集合主要集中在下面几个命名空间中:

1.1、集合的起源:接口关系


集合接口特点/说明IEnumerator、IEnumerator枚举器(还不是集合),提供foreach枚举项的能力IEnumerable、IEnumerable可枚举集合,几乎所有集合都实现了该接口,属于集合最基础的接口。就一个IEnumerator GetEnumerator() 方法,返回一个枚举器。ICollection、ICollection提供了基础集合操作:Count、Add()、Remove()、Clear()、Contains()、CopyTo()IList、IList索引器、IndexOf()、Insert()、RemoveAt()IDictionary、IDictionary键值集合操作:Keys、Values、索引器、Add()、Remove()IReadOnly***只读的集合,包括IReadOnlyCollection、IReadOnlyList、IReadOnlyDictionary等

[*]天赋技能 —— foreach:几乎所有集合都可以用foreach循环操作,是因为他们都继承自IEnumerable接口,由枚举器(IEnumerator)提供枚举操作。
[*]几乎所有集合都提供添加、删除、计数,来自基础接口 ICollection、ICollection。
[*]IList、IList 提供了数组的索引器、查找、插入等操作,几乎所有具体的集合类型都实现了该接口。
[*]Array 是一个抽象类,是所有数组T[]的基类,她是类型安全的。
[*]推荐尽量使用数组T[]、泛型版的集合,提供了更好的类型安全和性能。

1.2、非泛型集合—— 还有什么存在的价值?


[*]非泛型的Hashtable,Key、Value都是Object类型的,Dictionary 是泛型版本的 Hashtable。
[*]ArrayList 是非泛型版本的 List,基本很少使用,也尽量不用。
❓既然非泛型版本类型不安全,性能还差,为什么还存在呢?
主要是历史原因,泛型是.Net2.0 引入的,因此为了向后兼容,依然保留的非泛型版本集合。在接口实现时,非泛型接口一般都是显示实现的,因此基本不会用到。不过在有些场景下,非泛型接口、集合还是有点用的,如类型不固定的集合,或者用接口作为约束条件或类型判断。
ArrayList arr = new ArrayList();
arr.Add(1);
arr.Add("sam");
arr.Add(new Point());
if (arr is IList) {}

class User<T> where T :IList {}1.3、Collection、List有何不同?

❓两者比较相似,他们到底有什么区别呢?该如何选择?

[*]Collection 作为自定义集合基类,内部提供了一些virtual的实现,便于继承实现自己的集合类型。其内部集合用的就是List,如下部分源码 Collection.cs。
[*]List 作为集合使用,是最常用的可变长集合类型了,他优化了性能,但是丢失了可扩展性,没有提供任何可以override的成员。
public class Collection<T>
{
    public Collection()
    {
      items = new List<T>();
    }
    protected virtual void InsertItem(int index, T item)
    {
      items.Insert(index, item);
    }
}02、枚举器——foreach的秘密!

foreach 用来循环迭代可枚举对象,用一种非常简洁、优雅的姿势访问可枚举元素。常用于数组、集合,当然不仅限于集合,只要符合要求枚举要求的都可以。
foreach 可枚举类型说明数组包括Array数组、List、字典等,他们都实现了IEnumerable接口的。IEnumerable可枚举接口IEnumerable同上,泛型版本GetEnumerator()方法包含公共方法“IEnumerator GetEnumerator();”的任意类型yield迭代器yield语句实现的迭代器,实际返回的也是IEnumerable、IEnumerator

2.1、IEnumerator枚举器

枚举可以foreach 枚举的密码是他们都继承自IEnumerable接口,而更重要的是其内部的枚举器 —— IEnumerator。枚举器IEnumerator定义了向前遍历集合元素的基本协议,其申明如下:
public interface IEnumerator
{
        object Current { get; }

        bool MoveNext();

        void Reset();//这个方法是非必须的,用于重置游标,可不实现
}
public interface IEnumerator<out T> : IDisposable, IEnumerator
{
        new T Current { get; }
}

[*]MoveNext() 移动当前元素到下一个位置,Current获取当前元素,如果没有元素了,则MoveNext()返回false。注意MoveNext()会先调用,因此首次MoveNext()是把位置移动到第一个位置。
[*]Reset()用于重置到起点,主要用于COM互操作,使用很少,可不用实现(直接抛出 NotSupportedException)。
<blockquote>

来源:https://www.cnblogs.com/anding/p/18229596
免责声明:由于采集信息均来自互联网,如果侵犯了您的权益,请联系我们【E-Mail:cb@itdo.tech】 我们会及时删除侵权内容,谢谢合作!
页: [1]
查看完整版本: C#.Net筑基-集合知识全解