class Program

{

static void Main(string[] args)

{

Console.WriteLine("***第一种情况************************************************");

string[] array11 = new string[] { "111", "222", "333" };

string[] array12 = new string[] { "111", "222", "333" };

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

{

Console.WriteLine(array11[i]);

}

Console.WriteLine("=========================");

foreach (string s in array12)

{

Console.WriteLine(s);

}

/*结果

111

222

333

=========================

111

222

333

*/

//都是根据length来对数据进行循环

//不同的是foreach里自动对array进行index + 1的操作来循环,而for则是自己的代码控制的

Console.WriteLine("***第二种情况************************************************");

string[] array21 = new string[] { "111", "222", "333" };

string[] array22 = new string[] { "111", "222", "333" };

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

{

Console.WriteLine(array21[i]);

array21 = new string[] { "AAA", "BBB", "CCC" };

}

Console.WriteLine("=========================");

foreach (string s in array22)

{

Console.WriteLine(s);

array22 = new string[] { "AAA", "BBB", "CCC" };

}

/*结果

111

BBB

CCC

=========================

111

222

333

*/

//不一样了吧,看来在foreach内部的循环中对 源数据的更改 不是即时生效的

//在foreach(...)循环里尽量不要更改 操作的源数据

Console.ReadKey();

}

}

namespace for_foreach2

{

class Program

{

static void Main(string[] args)

{

//使用测试类E,做for和foreach的循环

string[] array31 = new string[] { "111", "222", "333" };

E e = new E(array31);

foreach (string s in e)

{

Console.WriteLine(s);

}

/*result

333

222

111

Dispose here!

*/

//差异出现了,这次是按照倒序的方式,而且还自动调用了Dispose方法!

Console.ReadKey();

}

}

//根据MS的参考,能在foreach上做循环的只能是实现了IEnumerable接口。

//(事实上,System.Array也是实现了IEnumerable接口的)

//这儿做一个在IEnumerable上的循环

public class E : System.Collections.IEnumerable

{

private InnerEnumerator inner;

public E(string[] array)

{

this.inner = new InnerEnumerator(array);

}

#region IEnumerable Members

public IEnumerator GetEnumerator()

{

return this.inner;

}

#endregion

private class InnerEnumerator : IEnumerator, IDisposable

{

private string[] s;

private int currentIndex;

public InnerEnumerator(string[] array)

{

this.s = array;

this.Reset();

}

#region IEnumerator Members

//Reset index to original(重置索引为原始的)

public void Reset()

{

this.currentIndex = s.Length - 1;

}

//Get Current object inner

public object Current

{

get

{

object o = this.s[this.currentIndex];

this.currentIndex--;

return o;

}

}

//Is there has any other object in the array?

public bool MoveNext()

{

if (this.currentIndex < 0)

{

return false;

}

return true;

}

#endregion

#region IDisposable Members

//Dispose Here()

public void Dispose()

{

Console.WriteLine("Dispose here!");

}

#endregion

}

}

}

综合上面,得出如下结论:
1.for循环并不依赖于数组或其他形式的组式数据结构,只是简单的
在调用了代码后,进行一个判断,判断是否要继续。
(非常类似于do..while和while循环--在这里不作具体分析了^_^~~)
2.foreach循环如果作用在一个基于System.Array的类型之上的数组的话,编译器会自动优化成与for循环非常类似
的代码,只是调用的指命有细微的差别,并且检查(包括编译阶段和运行时)会比for严格的多
3.foreach循环作用在一个非System.Array类型上(且一定要是实现了IEnumerable接口的类),会先调用
IEnumerable.GetEnumerator()方法获取一个Enumertor实例,再在获取的Enumertor实例上调用
GetCurrent()和MoveNext()方法,最后判断如果Enumertor实例如果实现了IDispose接口,就自动调用
IDispose.Dispose()方法!

那么我们应该分别在那些地方用for和foreach捏
建议:
1.在有对所循环的本体(System.Array)做赋值操作时,尽量不要用Foreach()。
2.foreach比for更灵活。(可在MoveNext()和GetCurrent()里编写自己的代码).
自己编写的类如果实现了IEnumerable接口的话,就可以用foreach循环了,而不管内部是否有一个真实的数组,
并且可以自定义循环的规则。
3.从OO的原则看,foreach循环更适于多数情况的使用
(事实上,foreach的实现是典型的Iterator模式,下面有简单的描述它的好处)
想用统一的调用循环接口时,foreach是最佳的选择
(MS有很多类就是这样的,例如前面提到的DataRowCollection.)

参考至:C#里for和foreach的区别 - lyh830612的日志 - 网易博客

黑马程序员-for和foreach的更多相关文章

  1. 黑马程序员——【Java高新技术】——代理

    ---------- android培训.java培训.期待与您交流! ---------- 一.“代理概述”及“AOP概念” (一)代理概述 1.问题:要为已存在的多个具有相同接口的目标类的各个方法 ...

  2. 黑马程序员+SQL基础(上)

    黑马程序员+SQL基础 ---------------<a href="http://edu.csdn.net"target="blank">ASP ...

  3. 黑马程序员+Winform基础(上)

    黑马程序员+Winform基础 ---------------<a href="http://edu.csdn.net"target="blank"> ...

  4. 黑马程序员—C语言的函数、数组、字符串

    ------Java培训.Android培训.iOS培训..Net培训.期待与您交流! ------- 一.函数 定义:函数是代码复用的一种机制. 函数的基本语法: 返回类型 函数名 ( 参数类型 参 ...

  5. 黑马程序员:Java基础总结----泛型(高级)

    黑马程序员:Java基础总结 泛型(高级)   ASP.Net+Android+IO开发 . .Net培训 .期待与您交流! 泛型(高级) 泛型是提供给javac编译器使用的,可以限定集合中的输入类型 ...

  6. 黑马程序员:Java基础总结----类加载器

    黑马程序员:Java基础总结 类加载器   ASP.Net+Android+IO开发 . .Net培训 .期待与您交流! 类加载器 Java虚拟机中可以安装多个类加载器,系统默认三个主要类加载器,每个 ...

  7. 黑马程序员:Java基础总结----静态代理模式&动态代理

    黑马程序员:Java基础总结 静态代理模式&动态代理   ASP.Net+Android+IO开发 . .Net培训 .期待与您交流! 静态代理模式 public  class  Ts {   ...

  8. 黑马程序员:Java基础总结----枚举

    黑马程序员:Java基础总结 枚举   ASP.Net+Android+IO开发 . .Net培训 .期待与您交流! 枚举 为什么要有枚举 问题:要定义星期几或性别的变量,该怎么定义?假设用1-7分别 ...

  9. 黑马程序员:Java基础总结----反射

    黑马程序员:Java基础总结 反射   ASP.Net+Android+IO开发 . .Net培训 .期待与您交流! 反射 反射的基石:Class类 Class类代表Java类,它的各个实例对象又分别 ...

随机推荐

  1. MiniProfiler.3.0.10 用于MVC4.0中不能显示SQL语句

    MiniProfiler.3.0.10 用于MVC4.0中可以显示执行时间,但是不能显示SQL语句,怎么解决?

  2. hdu 5150 Sit sit sit

    http://acm.hdu.edu.cn/showproblem.php?pid=5151 直接判断是不是素数,然后再注意1就行. #include <cstdio> #include ...

  3. XFS文件系统功能解析

    XFS文件系统是作为一个日志文件系统开发,采用B-树平衡树算法来尽快地分配数据.主要的设计目的之一是支持大型文件和大型文件系统.当前,能够支持的最大文件大小是2艾字节,最大文件系统大小为8艾字节. X ...

  4. JButton 做图片框

    JButton setHorizontalTextPosition(SwingConstants.CENTER);// 在水平方向文字位于图片中央 setVerticalTextPosition(Sw ...

  5. bzoj2818

    我们先穷举素数p然后令y>x 这样问题就是求这个gcd(x,y)=p  (1<=x<y=n)不难发现必须y=kp k∈N* 当y=p时,易知个数为φ(1)当y=2p 个数为φ(2), ...

  6. 「Poetize4」上帝造题的七分钟2

    描述 Description "第一分钟,X说,要有数列,于是便给定了一个正整数数列.第二分钟,L说,要能修改,于是便有了对一段数中每个数都开平方(下取整)的操作.第三分钟,k说,要能查询, ...

  7. jquery 学习第一课之start

    1.$选取符 ( $ == jQuery ) (1) $("div").addClass("special");选取本页面中的所有<div>元素,然 ...

  8. 【转】设置SecureCRT会话的缓冲区大小

    原文网址:http://guanjianfeng.com/archives/1484 在使用SecureCRT操作设备时,默认的回滚行数为500行.可以通过打开[选项]->[会话选项]-> ...

  9. 【动态规划】Vijos P1104 采药(NOIP2005普及组第三题)

    题目链接: https://vijos.org/p/1104 题目大意: T时间,n个物品,每个耗时ti,可获得收益ci,求最大收益. 题目思路: [动态规划] 01背包裸题.一维二维都过了,放个一维 ...

  10. cobbler客户端重装系统

    已有操作系统的主机通过koan从Cobbler服务器重装系统 1,安装epel rpm -ivh http://dl.fedoraproject.org/pub/epel/6/x86_64/epel- ...