`
persistC
  • 浏览: 71357 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

5.0for循环

阅读更多
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 = integers[j];
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循环,会在编译期间被看成是这样的形式:

  清单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 Random[]{new 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>(全部元素都是String的Collection)等。

  清单10:使用要被遍历的对象中的元素的上级类型的循环变量

String[] strings = {"A", "B", "C", "D"};
Collection<String> 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<Byte>,也是可行的。

  在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<T>”。

  在所有出现java.util.Iterator的地方,都写成“Iterator<T>”。

  在实现java.util.Iterator的接口的时候,用T作为next()方法的返回值类型。

  注意,这里的T不能是一个基本类型。如果打算用基本类型作为循环变量,那么得用它们的包裹类来代替这里的T,然后借助Auto-Unboxing机制,来近似的达到目的。

  清单13:用int型的循环变量来遍历一个能取出10个Integer元素的类

import java.util.*;
public class TenIntegers implements Iterable<Integer> {
 public Iterator<Integer> iterator() {
  return new Iterator<Integer>() {
   private int count = 0;
   public boolean hasNext() {
    return (count < 10);
   }
   public Integer next() {
    return Integer.valueOf(count++);
   }
   public void remove() {
    throw new UnsupportedOperationException();
   }
  };
 }
 public static void main(String[] args)
 {
  TenIntegers integers = new TenIntegers();
  for (int i : integers)
  {
   System.out.println(i);/* 依次输出从“0"到“9”的十个整数 */
  }
 }
}
  另外,一个类只能实现一次java.lang.Iterable接口,即使在后面的尖括号里使用不同的类型。类似“class A implements Iterable<String>, Iterable<Integer>”的写法,是不能通过编译的。所以,没有办法让一个可遍历对象能在这样遍历时,既可以使用Integer,又可以使用String来作为循环变量的类型(当然,把它们换成另外两种没有继承和自动转化关系的类也一样行不通)。

  清单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>来表示全部元素类型都是T的Collection,如Collection<String>、Collection<Integer>等。不过这里的T不能是一个简单类型,象Collection<int>之类的写法是不被认可的。

  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的资格问题

  6. 归纳总结

  借助J2SE 1.5中引入的第二种for循环,可以用一种更简单地方式来完成遍历。能用这种方法遍历的对象的类型,可以是数组、Collection或者任何其它实现了java.lang.Iterable接口的类。通过跟同样是在J2SE 1.5中引入的泛型机制配合使用,可以精确的控制能采用的循环变量的类型。而且,因为这么编写的代码,会在编译期间被自动当成是和传统写法相同的形式,所以不必担心要额外付出性能方面的代价。
分享到:
评论
1 楼 jinbo0120 2009-06-12  
受教了,我还一直存在疑问呢,原来是1.5开始使用的...看来我落伍了

相关推荐

    JDK1.5(5.0)增强循环的讲解[.txt]

    用 for in 在 Java 5.0 中增强循环 讲解十分详细可读性很强~~

    jdk5.0新特性ppt

    jdk5.0新特性 包括泛型,增强的for循环,自动的封箱和拆箱

    快速学习C语言中for循环语句的基本使用方法

    对于某个特定任务我们可以采用多种方法来编写程序。下面这段代码也可以实现前面的温度转换程序的功能:#include /*打印华氏温度—摄氏温度对照表*/ main() ...for 语句是一种循环语句,它是对 while 语

    C#5.0本质论第四版(因文件较大传的是百度网盘地址)

    3.7.2 for循环 86 3.7.3 foreach循环 88 3.7.4 switch语句 90 3.8 跳转语句 92 3.8.1 break语句 92 3.8.2 continue语句 94 3.8.3 goto语句 95 3.9 C#预处理指令 97 3.9.1 排除和...

    ACCP 5.0使用Java语言理解程序逻辑

    《for循环》 在解决有固定循环次数问题时,首选for循环结构

    Thinkphp5.0框架视图view的循环标签用法示例

    主要介绍了Thinkphp5.0框架视图view的循环标签用法,结合实例形式分析了thinkPHP5框架视图view中的volist标签、foreach标签、for标签相关使用方法,需要的朋友可以参考下

    练习3-1 创建一个自动索引的数组.vi

    LabVIEW清华题库练习3-1 创建一个自动索引的数组.vi

    jdk5.0新特性汇总(超级详细)

    加强for循环,静态导入,枚举,泛型,反射,注解,类加载器,代理等。张孝祥总结,绝对经典。

    笔记分享5.0 三种程序结构1

    第五章 三种程序结构 章节时长:25min 第一节 判断语句:if语句 8min第二节 循环语句1:while循环 3.5min第三节 循环语句2:for循

    Accp5.0 Y2用各种方法做的网上订餐系统

    用for循环 Map JSRL和EL Servlet等多种方法实现的网上订餐系统 可供大家参考,并在里面实现了购物车中数量的更新(即购买数量可用输入的方式购买多个商品)

    jdk1.5的新特性

    从增强的for循环到诸如泛型(generic)之类更复杂的特性,都将很快出现在您所编写的代码中。我们刚刚完成了一个基于Java 5.0的大型任务,而本文就是要介绍我们使用这些新特性的体验。本文不是一篇入门性的文章,而是对...

    JDK5.0的11个主要新特征.doc

    2 增强循环(Enhanced for Loop) 3 可变参数(Variable Arguments) 4 自动实现装箱和解箱操作(Boxing/Unboxing Conversions) 5 静态导入(Static Imports) 6 枚举类(Enumeration Classes) 7 元数据(Meta data) 8 ...

    信息技术考试卷-python图文练习.doc

    如下Python程序段 for i in range(1,4): for j in range(0,3): print ("Python") 语句print ("Python")的执行次数是( ) A.3 B.4 C.6 D.9 5.关于python程序设计语言,下列说法错误的是( ) A....

    JAVA中的二维数组

    //方法二:使用for循环动态赋值 //导入java.util.*;包 Scanner input=new Scanner(System.in); int [] arr=new int[5]; for(int i=0;i;i++) { System.out.println("请输入第"+(i+1)+"个值"); arr[i]=...

    to_alice.c

    主要用到的数据结构是一个结构体数组,通过使用for循环去把乐谱从结构体数组里面提取出来,然后switch case的嵌套获取音符,音调,通过操作结构体变量获取音符持续时间,把各种数据通过传参,使用驱动的ioctl接口去...

    试卷试卷2Python三级考试练习题word复习知识点试卷试题.pdf

    s=0 for i in range(1,5): s=s+i print("s=",s) A.s=5 B.s=6 C.s=10 D.s=15 2.在 python 中,以下哪个选项 a 的数据类型为整型( ) A.a=5 B.a=input() C.a='5' D.a=5.0 3.下列定义变量的 python 程序...

    java采购管理系统源码-retrotranslator:Retrotranslator是使Java应用程序兼容Java1.4、Java1.3

    循环 可变参数 协变返回类型 格式化输出 静态导入 并发实用程序 集合框架增强 下载 () 并解压二进制分发文件Retrotranslator-_n.n.n_-bin.zip ,其中nnn是最新的 Retrotranslator 版本号。 使用 Java 5.0 或 Java 6 ...

    Android代码-BluetoothLELibrary

    可以连续操作发送数据、读取特征、开启通知,即使你在for循环中写也没问题,自带队列。 扫描操作支持-&gt; 设置扫描时长、根据设备名称扫描、根据硬件地址扫描、根据服务UUID扫描、连接成功后自动关闭扫描。 队列定时...

    tengge py for s60编程教程

    python for s60编程最适合机型为塞班s60v3横屏手机。请自行在手机上安装python平台, 代码编写编译反编译测试工具“ipro7”,请安装sis程序打包解包py程序打包权限修改 签名工具“智能解包”。 其中包括: 1....

    C# for CSDN 乱七八糟的看不懂

    循环 无条件循环 int sum,x; sum=0; for(x=1;x;x++) { sum+=x; } 有条件循环 private void button1_Click(object sender, System.EventArgs e) { int sum=0; int x=0; while ((sum) & (x)) { x++; sum+=x; } string...

Global site tag (gtag.js) - Google Analytics