弱类型集合和强类型集合

本文深入探讨了.NET Framework中的集合类,包括ArrayList、Queue、Stack及哈希表等,讲解了它们的基本用法和特点,以及如何进行强类型集合的创建。

上次我们看到的StaffCollection和DepartmentCollection都是继承于System.Collections.CollectionBase,那CollectionBase是个怎样的类呢?
我们知道数组的定义是:连续分配的内存空间,大小不能被改变,因此如果我们需要一个像单向链表一样可以动态添加元素的数据结构,我们就需要使用集合。

System.Collections.ArrayList一个动态的数组,从严格意义上讲,说它是数组是不合适的,因为ArrayList一个重要的特征是,可以放入各种对象,该List是一个object的容器。
System.Collections.ArrayList提供了一组常用的属性和方法:

属性

方法

可包含的元素Capacity添加对象Add、AddRange、Insert、InsertRange

实际包含的元素

Count将容量设置为 ArrayList 中元素的实际数量TrimToSize
是否具有固定大小IsFixedSize移除元素Clear、Remove、RemoveAt、RemoveRange
是否为只读IsReadOnly 匹配搜索IndexOf、LastIndexOf
获取或设置指定索引处的元素Item排序Sort
将 ArrayList 的元素复制到新数组中ToArray

1None.gifSystem.Collections.ArrayListlist=newSystem.Collections.ArrayList();
2None.giflist.Add("hello");
3None.giflist.Add(123);
4None.giflist.Add(DateTime.Now);
5None.giffor(inti=0;i<=list.Count-1;i++)
6ExpandedBlockStart.gifContractedBlock.gifdot.gif{
7InBlock.gifSystem.Console.WriteLine(list[i]);
8ExpandedBlockEnd.gif}

以上的代码运行起来,似乎没有什么奇特,屏幕上显示出
hello
123
2007-2-13 8:54:12
不过,再进一步分析,list[i]所返回的都是object类型。也就是说System.Console.WriteLine(list[i]);其实是System.Console.WriteLine(list[i].ToString());的简略写法。
你不可能通过list[2]返回一个DateTime类型,并对该类型作日期加减处理(见前面的日期处理章节)。要返回真正的类型,必须要做类型转换。
我们来看看System.Console.WriteLine(list[i].GetType());的输出
System.String
System.Int32
System.DateTime
我们来做一下变化
1None.giffor(inti=0;i<=list.Count-1;i++)
2ExpandedBlockStart.gifContractedBlock.gifdot.gif{
3InBlock.gifswitch(list[i].GetType().ToString())
4ExpandedSubBlockStart.gifContractedSubBlock.gifdot.gif{
5InBlock.gifcase"System.String":
6InBlock.gifSystem.Console.WriteLine(((string)list[i])+"C#");
7InBlock.gifbreak;
8InBlock.gifcase"System.Int32":
9InBlock.gifSystem.Console.WriteLine(((int)list[i])*4);
10InBlock.gifbreak;
11InBlock.gifcase"System.DateTime":
12InBlock.gifSystem.Console.WriteLine(((DateTime)list[i]).AddDays(15));
13InBlock.gifbreak;
14ExpandedSubBlockEnd.gif}

15ExpandedBlockEnd.gif}


现在的运行结果就成了
hello C#
492
2007-2-28 9:01:17

如果返回值是object类型的,就称为弱类型,否则就是强类型。
所以说,ArrayList是一个弱类型的集合,该集合中可以放置所有的数据类型。这是不很安全。
因此我们往往需要自己编写一个强类型的集合
.Net提供了一个用于编写强类型集合的基类:System.Collections.CollectionBase
该类公开了集合中常用的,但和类型无关的操作:
Count、Clear 、RemoveAt
最最主要的是,它提供了一个protected的InnerList,该InnerList其实就是ArrayList,因此我们自己做的强类型集合,其实就是对ArrayList的包装。昨天的代码是对CollectionBase提供的protected的List操作,该List是一个IList接口实现(IList是List的返回接口),功能比InnerList弱,如果你现在把昨天的代码修改为使用InnerList,你会发现,你代码可以少写的多。InnerList返回的是ArrayList的实例对象。


除了ArrayList集合,还有:

队列Queue表示对象的先进先出集合

1None.gifSystem.Collections.Queueq=newSystem.Collections.Queue();
2None.giffor(inti=0;i<5;i++)
3ExpandedBlockStart.gifContractedBlock.gifdot.gif{
4InBlock.gifq.Enqueue(i);
5ExpandedBlockEnd.gif}

6None.gifSystem.Console.WriteLine(q.Peek());
7None.gifwhile(q.Count>0)
8ExpandedBlockStart.gifContractedBlock.gifdot.gif{
9InBlock.gifSystem.Console.WriteLine(q.Dequeue());
10ExpandedBlockEnd.gif}

11None.gifSystem.Console.ReadLine();

代码运行显示
0
0
1
2
3
4
为什么有两个0呢?因为Peek 返回位于 Queue 开始处的对象后不会移除该对象。


和Queue相反的是Stack,对象的简单的后进先出集合(不是后进后出哦)
因此,代码

1None.gifSystem.Collections.Stacks=newSystem.Collections.Stack();
2None.giffor(inti=0;i<5;i++)
3ExpandedBlockStart.gifContractedBlock.gifdot.gif{
4InBlock.gifs.Push(i);
5ExpandedBlockEnd.gif}

6None.gif
7None.gifs.Peek();
8None.gifwhile(s.Count>0)
9ExpandedBlockStart.gifContractedBlock.gifdot.gif{
10InBlock.gifSystem.Console.WriteLine(s.Pop());
11ExpandedBlockEnd.gif}

运行的结果是
4
3
2
1
0
集合还有一个很重要的对象:哈希表,表示键/值对的集合。这个集合就是一个两列的两维表,第一列表示key,键是不可以重复的。
你可以检查,
是否包含特定键:ContainsKey
是否包含特定值:ContainsValue

1None.giftable.Add("Hello","HELLO");
2None.giftable.Add("HELLO","HELLO");
3None.giftable.Add("hello","HELLO");

取值的时候,是根据键名称取值
System.Console.WriteLine(table["hello"]);

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值