JAVA for 和foreach

结论:

for更高效、foreach更安全
在遍历过程中有可能对数据进行修改的情况下使用foreach
只在线程内的临时变量遍历就使用for

原因:

foreach是通过GetEnumerator获得一个IEnumerator对象,通过IEnumerator对象执行MoveNext()方法和获取Current属性进行遍历的。如下两种写法,编译得到的代码相同。

//第一种写法:  
IList list = new ArrayList();  
IEnumerator iter = list.GetEnumerator();  
try  
{  
    while (iter.MoveNext())  
    {  
        Object obj = iter.Current;  
        //do something ...  
    }  
}  
finally  
{  
    IDisposable disposableObj = iter as IDisposable;  
    if (disposableObj != null)  
    {  
        disposableObj.Dispose();  
    }  
}  
  
//第二种写法:  
IList list = new ArrayList();  
foreach (Object obj in list)  
{  
    //do something ...  
}  

查看java源码:

public class ArrayList  
{  
    //这是一个版本标识,ArrayList对象,每做一个修改操作,_version都会加1  
    private int _version;  
  
    public virtual int Add(object value)  
    {  
        int num1;  
        if (this._size == this._items.Length)  
        {  
            this.EnsureCapacity((this._size + 1));  
        }  
        this._items[this._size] = value;  
        ++this._version; //注意此处  
        this._size = ((num1 = this._size) + 1);  
        return num1;  
    }  
  
    public virtual void Clear()  
    {  
        Array.Clear(this._items, 0, this._size);  
        this._size = 0;  
        ++this._version; //注意此处  
    }  
  
    //每次调用GetEnumerator方法,都会构造一个FastArrayListEnumerator  
    //或者ArrayListEnumeratorSimple对象。  
    public virtual IEnumerator GetEnumerator()  
    {  
        if (base.GetType() == typeof(ArrayList))  
        {  
            return new ArrayList.FastArrayListEnumerator(this);  
        }  
        return new ArrayList.ArrayListEnumeratorSimple(this);  
    }  
}  

class FastArrayListEnumerator  
{  
    private int version;  
  
    internal FastArrayListEnumerator(ArrayList list)  
    {  
        this.list = list;  
        this.index = -1;  
  
        //获取构建FastArrayListEnumerator对象时ArrayList的版本号  
        this.version = list._version;   
  
        this.lastIndex = (list._size - 1);  
    }  
  
    public bool MoveNext()  
    {  
        int num1;  
  
        //比较ArrayList当前的版本号,  
        //是否和构建FastArrayListEnumerator对象时的版本号一致  
        //如果不一致,则抛出异常。  
        if (this.version != this.list._version)  
        {  
            throw new InvalidOperationException(  
                Environment.GetResourceString("InvalidOperation_EnumFailedVersion")  
                );  
        }  
  
        //... ...   
    }  
} 

rrayList是通过_version成员变量作版本标识的,每次执行Add、Clear等修改ArrayList内容的操作,都会将版本号加1,而每次调用GetEnumerator方法,都会构造一个FastArrayListEnumerator或者ArrayListEnumeratorSimple对象。
FastArrayListEnumerator对象构建时,当时时ArrayList的版本号。当执行MoveNext()操作时,检查ArrayList当前的版本号是否和FastArrayListEnumerator对象构建时的版本号一致,如果不一致就会抛出异常。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 传送门 解读阿里Java开发手册(v1.1.1) - 异常日志 前言 阿里Java开发手册谈不上圣经,但确实是大量...
    kelgon阅读 4,375评论 4 50
  • (一)Java部分 1、列举出JAVA中6个比较常用的包【天威诚信面试题】 【参考答案】 java.lang;ja...
    独云阅读 7,136评论 0 62
  • java笔记第一天 == 和 equals ==比较的比较的是两个变量的值是否相等,对于引用型变量表示的是两个变量...
    jmychou阅读 1,516评论 0 3
  • https://zhuanlan.zhihu.com/p/24687801?refer=dreawer http:...
    待汝豪杰只是凡夫阅读 286评论 0 0
  • 郦波和蒋勋老师的声音都很有磁性,听他们讲中国古典文学作品真是一种莫大的享受。刚才郦波老师讲《牡丹亭》时提到了一个很...
    写下即永恒阅读 707评论 0 0