分享
 
 
 

在Visual C# 2.0中创建优雅代码

王朝c#·作者佚名  2006-11-24
窄屏简体版  字體: |||超大  

热衷于C#语言的人会喜欢上Visual C# 2005。Visual Studio 2005为Visual C# 2005带来了大量令人兴奋的新功能,例如泛型、迭代器、局部类和匿名方法等。虽然泛型是人们最常谈到的也是预期的功能,尤其是在熟悉模板的C++开发人员中间,但是其他的新功能同样是对Microsoft .NET开发宝库的重要补充。与C#的第一个版本相比,这些功能和语言附加将会提高整体的生产效率,从而使开发人员能够以更快的速度写出更加简洁的代码。

迭代器

在C# 1.1中,可以使用foreach循环来遍历诸如数组、集合这样的数据结构:

string[] cities = {"New York","Paris","London"};

foreach(string city in cities)

{

Console.WriteLine(city);

}

实际上,可以在foreach循环中使用任何自定义数据集合,只要该集合类型实现了返回IEnumerator接口的GetEnumerator方法即可。通常,需要通过实现IEnumerable接口来完成这些工作:

public interface IEnumerable

{

IEnumerator GetEnumerator();

}

public interface IEnumerator

{

object Current{get;}

bool MoveNext();

void Reset();

}

在通常情况下,实现IEnumerable接口的类是作为要遍历的集合类型的嵌套类来提供的。这样,此种迭代器设计模式维持了迭代的状态。将嵌套类作为枚举器的好处是因为它可以访问其包含类的所有私有成员,而且,对迭代客户端隐藏了底层数据结构的实际实现细节,使得能够在多种数据结构上使用相同的客户端迭代逻辑,如图1所示。

图1 迭代器设计模式

此外,由于每个迭代器都保持单独的迭代状态,所以多个客户端可以执行单独的并发迭代。通过实现IEnumerable接口,诸如数组和队列这样的数据结构可以支持这种非常规的迭代。在foreach循环中生成的代码调用类的GetEnumerator方法可以简单地获得一个IEnumerator对象,然后将其用于while循环,接着,通过连续调用它的MoveNext方法来遍历集合。如果您需要显式地遍历集合,您可以直接使用IEnumerator(无须使用foreach语句)。

但是使用这种方法有一些问题。首先,如果集合包含值类型,则需要对它们进行装箱和拆箱才能获得项,因为IEnumerator.Current返回一个Object类的对象。这将导致潜在的性能降低 和托管堆上的压力增大。即使集合包含引用类型,仍然会产生从对象向下强制类型转换的不利结果。虽然大多数开发人员不熟悉这一特性,事实上在C# 1.0中,不必实现IEnumerator或IEnumerable接口就可以为每个循环实现迭代器模式。编译器将选择调用强类型化版本,以避免强制类型转换和装箱。结果是,即使在1.0版本中,也可能没有导致性能损失。

为了更好地阐明这个解决方案并使其易于实现,Microsoft .NET框架2.0在System.Collections.Generics命名空间中定义了类型安全的泛型IEnumerable<ItemType>和IEnumerator<ItemType>:

public interface IEnumerable

{

IEnumerator GetEnumerator();

}

public interface IEnumerator : IDisposable

{

ItemType Current{get;}

bool MoveNext();

}

除了利用泛型之外,新的接口与其前身还略有差别。与IEnumerable不同,IEnumerator是从IDisposable派生而来的,并且没有Reset方法。图2中的代码显示了实现IEnumerable<string>的简单city集合,而图3显示了编译器展开foreach循环的代码中如何使用该接口。图2中的实现使用了名为MyEnumerator的嵌套类,它将一个引用作为构造参数返回给要枚举的集合。MyEnumerator清楚地知道city集合(本例中的一个数组)的实现细节。此外,MyEnumerator类使用m_Current成员变量维持当前的迭代状态,此成员变量用作数组的索引。

public class CityCollection : IEnumerable<string>

{

string[] m_Cities = {"New York","Paris","London"};

public IEnumerator<string> GetEnumerator()

{

return new MyEnumerator(this);

}

//Nested class definition

class MyEnumerator : IEnumerator<string>

{

CityCollection m_Collection;

int m_Current;

public MyEnumerator(CityCollection collection)

{

m_Collection = collection;

m_Current = -1;

}

public bool MoveNext()

{

m_Current++;

if(m_Current < m_Collection.m_Cities.Length)

return true;

else

return false;

}

public string Current

{

get

{

if(m_Current == -1)

throw new InvalidOperationException();

return m_Collection.m_Cities[m_Current];

}

}

public void Dispose(){}

}

}

图2 实现IEnumerable<string>

CityCollection cities = new CityCollection();

//For this foreach loop:

foreach(string city in cities)

{

Trace.WriteLine(city);

}

//The compiler generates this equivalent code:

IEnumerable<string> enumerable = cities;

IEnumerator<string> enumerator = enumerable.GetEnumerator();

using(enumerator)

{

while(enumerator.MoveNext())

{

Trace.WriteLine(enumerator.Current);

}

}

图3 简单的迭代程序

第二个问题迭代器的实现也是难以解决的问题。虽然对于简单的应用实例中(如图3所示),实现是相当简单的,但是对于高级的数据结构,实现将非常复杂,例如二叉树,它需要递归遍历,并需在递归时维持迭代状态。另外,如果需要各种迭代选项,例如需要在一个链表中从头到尾和从尾到头选项,则此链表的代码就会因为使用多种迭代器实现而变得臃。这正是设计C# 2.0迭代器所要解决的问题。通过使用迭代器,可以让C#编译器生成IEnumerator的实现。C#编译器能够自动生成一个嵌套类来维持迭代状态。可以在泛型集合或特定于类型的集合中使用迭代器。开发人员需要做的只是告诉编译器在每个迭代中产生的是什么。如同手动提供迭代器一样,需要公开GetEnumerator方法,此方法是在实现IEnumerable接口或IEnumerable<ItemType>公开的。

可以使用新的C#的yield return语句告诉编译器产生什么。例如,下面的代码显示了如何在city集合中使用C#迭代器来代替图2中的人工实现部分:

public class CityCollection : IEnumerable<string>

{

string[] m_Cities = {"New York","Paris","London"};

public IEnumerator<string> GetEnumerator()

{

for(int i = 0; i<m_Cities.Length; i++)

yield return m_Cities[i];

}

}

此外,您还可以在非泛型集合中使用C#迭代器:

public class CityCollection : IEnumerable

{

string[] m_Cities = {"New York","Paris","London"};

public IEnumerator GetEnumerator()

{

for(int i = 0; i<m_Cities.Length; i++)

yield return m_Cities[i];

}

}

此外,还可以在如图4所示的在完全泛型(Fully Generic)集合中使用C#迭代器。当使用泛型集合和迭代器时,从声明的集合(本例中的string)中,编译器就可以检索到foreach循环内IEnumerable<ItemType>所用的特定类型:

LinkedList list = new LinkedList();

/* Some initialization of list, then */

foreach(string item in list)

{

Trace.WriteLine(item);

}

图4在普通链表中使用迭代程序

//K is the key, T is the data item

class Node<K,T>

{

public K Key;

public T Item;

public Node<K,T> NextNode;

}

public class LinkedList<K,T> : IEnumerable<T>

{

Node<K,T> m_Head;

public IEnumerator<T> GetEnumerator()

{

Node<K,T> current = m_Head;

 

[1] [2] [3] [4] [5] [6] 下一页

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有