乍一看,这个程序可能会在一次又一次的运行中,以相等的概率打印出 Pain、Gain 或 Main。看起来该程序会根据随机数生成器所选取的值来选择单词的第一个字母:0 选 M,1 选 P,2 选 G。但它实际上既不会打印 Pain,也不会打印 Gain。也许更令人吃惊的是,它也不会打印 Main,并且它的行为不会在一次又一次的运行中发生变化,它总是在打印 ain。这又是为什么呢?
多个问题,纠结在一起导致了这个问题:
Random.nextInt (int) 的规范描述是这样写的:“返回一个伪随机地、均等地分布在从 0(包括)到指定的数值(不包括)之间的一个 int 数值”[Java-API]。这意味着表达式 rnd.nextInt (2) 可能的取值只有 0 和 1,Switch 语句将永远也到不了 case2 分支,这表示程序将永远不会打印 Gain。nextInt 的参数应该是 3 而不是 2;
case 中没有任何 break 语句。不论 switch 表达式为何值,该程序都将执行其相对应的 case 以及所有后续的 case [JLS 14.11]。因此,尽管每一个 case 都对变量 word 赋了一个值,但是总是最后一个赋值胜出,覆盖了前面的赋值。最后一个赋值将总是最后一种情况(default),即 new StringBuffer {‘M’}。这表明该程序将总是打印 Main,而从来不打印 Pain 或 Gain;
在本例中,编译器会选择接受 int 的构造器,通过拓宽原始类型转换把字符数值’M’转换为一个 int 数值 77 [JLS 5.1.2]。换句话说,new StringBuffer (‘M’) 返回的是一个具有初始容量 77 的空的字符串缓冲区。该程序余下的部分将字符 a、i 和 n 添加到了这个空字符串缓冲区中,并打印出该字符串缓冲区那总是 ain 的内容。
那怎么样呢,修改如下:
/
private static Random rnd = new Random(); public static void main(String[] args) { System.out.println("PGM".charAt(rnd.nextInt(3)) + "ain"); }