J2SE 1.5中增强for循环详解.doc_第1页
J2SE 1.5中增强for循环详解.doc_第2页
J2SE 1.5中增强for循环详解.doc_第3页
J2SE 1.5中增强for循环详解.doc_第4页
J2SE 1.5中增强for循环详解.doc_第5页
已阅读5页,还剩6页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

-J2SE 1.5中增强for循环详解J2SE 1.5提供了另一种形式的for循环。借助这种形式的for循环,可以用更简单地方式来遍历数组和Collection等类型的对象。本文介绍使用这种循环的具体方式,说明如何自行定义能被这样遍历的类,并解释和这一机制的一些常见问题。在Java程序中,要“逐一处理”或者说,“遍历”某一个数组或Collection中的元素的时候,一般会使用一个for循环来实现(当然,用其它种类的循环也不是不可以,只是不知道是因为for这个词的长度比较短,还是因为for这个词的含义和这种操作比较配,在这种时候for循环比其它循环常用得多)。对于遍历数组,这个循环一般是采取这样的写法:清单1:遍历数组的传统方式/* 建立一个数组 */int integers = 1, 2, 3, 4;/* 开始遍历 */for (int j = 0; j integers.length; j+) int i = integersj;System.out.println(i);而对于遍历Collection对象,这个循环则通常是采用这样的形式:清单2:遍历Collection对象的传统方式/* 建立一个Collection */String strings = A, B, C, D;Collection stringList = java.util.Arrays.asList(strings);/* 开始遍历 */for (Iterator itr = stringList.iterator(); itr.hasNext();) Object str = itr.next();System.out.println(str);而在Java语言的最新版本J2SE 1.5中,引入了另一种形式的for循环。借助这种形式的for循环,现在可以用一种更简单地方式来进行遍历的工作。1. 第二种for循环不严格的说,Java的第二种for循环基本是这样的格式:for (循环变量类型 循环变量名称 : 要被遍历的对象) 循环体借助这种语法,遍历一个数组的操作就可以采取这样的写法:清单3:遍历数组的简单方式/* 建立一个数组 */int integers = 1, 2, 3, 4;/* 开始遍历 */for (int i : integers) System.out.println(i);/* 依次输出“1”、“2”、“3”、“4” */这里所用的for循环,会在编译期间被看成是这样的形式:1 (未取到网页内容)清单4:遍历数组的简单方式的等价代码/* 建立一个数组 */int integers = 1, 2, 3, 4;/* 开始遍历 */for (int 变量名甲 = 0; 变量名甲 integers.length; 变量名甲+) System.out.println(变量名甲);/* 依次输出“1”、“2”、“3”、“4” */这里的“变量名甲”是一个由编译器自动生成的不会造成混乱的名字。而遍历一个Collection的操作也就可以采用这样的写法:清单5:遍历Collection的简单方式/* 建立一个Collection */String strings = A, B, C, D;Collection list = java.util.Arrays.asList(strings);/* 开始遍历 */for (Object str : list) System.out.println(str);/* 依次输出“A”、“B”、“C”、“D” */这里所用的for循环,则会在编译期间被看成是这样的形式:清单6:遍历Collection的简单方式的等价代码/* 建立一个Collection */String strings = A, B, C, D;Collection stringList = java.util.Arrays.asList(strings);/* 开始遍历 */for (Iterator 变量名乙 = list.iterator(); 变量名乙.hasNext();) System.out.println(变量名乙.next();/* 依次输出“A”、“B”、“C”、“D” */这里的“变量名乙”也是一个由编译器自动生成的不会造成混乱的名字。因为在编译期间,J2SE 1.5的编译器会把这种形式的for循环,看成是对应的传统形式,所以不必担心出现性能方面的问题。不用“foreach”和“in”的原因Java采用“for”(而不是意义更明确的“foreach”)来引导这种一般被叫做“for-each循环”的循环,并使用“:”(而不是意义更明确的“in”)来分割循环变量名称和要被遍历的对象。这样作的主要原因,是为了避免因为引入新的关键字,造成兼容性方面的问题在Java语言中,不允许把关键字当作变量名来使用,虽然使用“foreach”这名字的情况并不是非常多,但是“in”却是一个经常用来表示输入流的名字(例如java.lang.System类里,就有一个名字叫做“in”的static属性,表示“标准输入流”)。的确可以通过巧妙的设计语法,让关键字只在特定的上下文中有特殊的含义,来允许它们也作为普通的标识符来使用。不过这种会使语法变复杂的策略,并没有得到广泛的采用。2. 防止在循环体里修改循环变量在默认情况下,编译器是允许在第二种for循环的循环体里,对循环变量重新赋值的。不过,因为这种做法对循环体外面的情况丝毫没有影响,又容易造成理解代码时的困难,所以一般并不推荐使用。Java提供了一种机制,可以在编译期间就把这样的操作封杀。具体的方法,是在循环变量类型前面加上一个“final”修饰符。这样一来,在循环体里对循环变量进行赋值,就会导致一个编译错误。借助这一机制,就可以有效的杜绝有意或无意的进行“在循环体里修改循环变量”的操作了。清单7:禁止重新赋值int integers = 1, 2, 3, 4;for (final int i : integers) i = i / 2; /* 编译时出错 */注意,这只是禁止了对循环变量进行重新赋值。给循环变量的属性赋值,或者调用能让循环变量的内容变化的方法,是不被禁止的。清单8:允许修改状态Random randoms = new Randomnew Random(1), new Random(2), new Random(3);for (final Random r : randoms) r.setSeed(4);/* 将所有Random对象设成使用相同的种子 */System.out.println(r.nextLong();/* 种子相同,第一个结果也相同 */3. 类型相容问题为了保证循环变量能在每次循环开始的时候,都被安全的赋值,J2SE 1.5对循环变量的类型有一定的限制。这些限制之下,循环变量的类型可以有这样一些选择:循环变量的类型可以和要被遍历的对象中的元素的类型相同。例如,用int型的循环变量来遍历一个int型的数组,用Object型的循环变量来遍历一个Collection等。清单9:使用和要被遍历的对象中的元素相同类型的循环变量int integers = 1, 2, 3, 4;for (int i : integers) System.out.println(i);/* 依次输出“1”、“2”、“3”、“4” */循环变量的类型可以是要被遍历的对象中的元素的上级类型。例如,用int型的循环变量来遍历一个byte型的数组,用Object型的循环变量来遍历一个Collection(全部元素都是String的Collection)等。清单10:使用要被遍历的对象中的元素的上级类型的循环变量String strings = A, B, C, D;Collection list = java.util.Arrays.asList(strings);for (Object str : list) System.out.println(str);/* 依次输出“A”、“B”、“C”、“D” */循环变量的类型可以和要被遍历的对象中的元素的类型之间存在能自动转换的关系。J2SE 1.5中包含了“Autoboxing/Auto-Unboxing”的机制,允许编译器在必要的时候,自动在基本类型和它们的包裹类(Wrapper Classes)之间进行转换。因此,用Integer型的循环变量来遍历一个int型的数组,或者用byte型的循环变量来遍历一个Collection,也是可行的。清单11:使用能和要被遍历的对象中的元素的类型自动转换的类型的循环变量int integers = 1, 2, 3, 4;for (Integer i : integers) System.out.println(i);/* 依次输出“1”、“2”、“3”、“4” */注意,这里说的“元素的类型”,是由要被遍历的对象的决定的如果它是一个Object型的数组,那么元素的类型就是Object,即使里面装的都是String对象也是如此。可以限定元素类型的Collection截至到J2SE 1.4为止,始终无法在Java程序里限定Collection中所能保存的对象的类型它们全部被看成是最一般的Object对象。一直到J2SE 1.5中,引入了“泛型(Generics)”机制之后,这个问题才得到了解决。现在可以用Collection来表示全部元素类型都是T的Collection,如Collection、Collection等。不过这里的T不能是一个简单类型,象Collection之类的写法是不被认可的。4. 被这样遍历的前提有两种类型的对象可以通过这种方法来遍历数组和实现了java.lang.Iterable接口的类的实例。试图将结果是其它类型的表达式放在这个位置上,只会在编译时导致一个提示信息是“foreach not applicable to expression type”的问题。java.lang.Iterable接口中定义的方法只有一个:iterator()返回一个实现了java.util.Iterator接口的对象而java.util.Iterator接口中,则定义了这样三个方法:hasNext()返回是否还有没被访问过的对象next()返回下一个没被访问过的对象remove()把最近一次由next()返回的对象从被遍历的对象里移除。这是一个可选的操作,如果不打算提供这个功能,在实现的时候抛出一个UnsupportedOperationException即可。因为在整个循环的过程中,这个方法根本没有机会被调用,所以是否提供这个功能,在这里没有影响。借助这两个接口,就可以自行实现能被这样遍历的类了。-。清单12:一个能取出10个Object元素的类import java.util.*;class TenObjects implements Iterable public Iterator iterator() return new Iterator() private int count = 0;public boolean hasNext() return (count 10);public Object next() return new Integer(count+);public void remove() throw new UnsupportedOperationException();public static void main(String args)TenObjects objects = new TenObjects();for (Object i : objects)System.out.println(i);/* 依次输出从“0到“9”的十个整数 */Collection的资格问题在J2SE 1.5的API中,所有能被这样遍历的对象的类型都是java.util.Collection的子类型,看上去很象java.util.Collection获得了编译器的特殊对待。不过,造成这种现象的实际原因,是在J2SE 1.5中,java.util.Collection被定义成了java.lang.Iterable的子接口。编译器并没有给Collection什么特别的关照。从理论上说,完全可以制造出一些拒不实现Collection接口的容器类,而且能让它们和Collection一样被用这种方法遍历。不过这样的容器类,可能会因为存在兼容性的问题,而得不到广泛的流传。若干方法的命名问题在java.lang.Iterable接口中,使用iterator(),而不是getIterator();而java.util.Iterator接口中,也使用hasNext()和next(),而不是hasNextElement()和getNextElement()。造成这种现象的原因,是Java Collections Framework的设计者们,认为这些方法往往会被频繁的调用(每每还会挤到一行),所以用短一点的名字更为合适。5. 加入更精确的类型控制如果在遍历自定义的可遍历对象的时候,想要循环变量能使用比Object更精确的类型,就需要在实现java.lang.Iterable接口和java.util.Iterator接口的时候,借助J2SE 1.5中的泛型机制,来作一些类型指派的工作。如果想要使循环变量的类型为T,那么指派工作的内容是:在所有要出现java.lang.Iterable的地方,都写成“Iterable”。在所有出现java.util.Iterator的地方,都写成“Iterator”。在实现java.util.Iterator的接口的时候,用T作为next()方法的返回值类型。注意,这里的T不能是一个基本类型。如果打算用基本类型作为循环变量,那么得用它们的包裹类来代替这里的T,然后借助Auto-Unboxing机制,来近似的达到目的。清单13:用int型的循环变量来遍历一个能取出10个Integer元素的类import java.util.*;public class TenIntegers implements Iterable public Iterator iterator() return new Iterator() private int count = 0;public boolean hasNext() return (coun

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论