题目1:Sets的乐趣程序
public class ShortSet{ public static void main(String args[]) { Set s = new HashSet(); for (short i=0; i<100; i++) { s.add(i); s.remove(i-1); } System.out.println(s.size()); }}
选项
(a) 1
(b) 100
(c) Throws exception
(d) None of the above
答案(b)100
分析过程
1 在Set中加入的是一个Short值,但是删除的却是Integer的值。(译者:从留言中,很高兴看到有些朋友看到了这一点)
2 此行程序,s.remove(i-1),中的i-1是一个返回值为int的指示。如果i是short或byte类型,当它与int或long做算术运算时,返回的一定是int或long类型的值。
3 在执行s.remove方法时,上面的int值将被autobox成一个Integer对象。
4 注意,包含值1的Short对象和包含值1的Integer对象在比较中不被认为是对等的。
5 在Set中加入Short对象后,再删除Integer对象,后者的命令是不会被运行的(no-op)。编译器也不会报错。
经验教训
1 在Java1.5中,Set的接口是这样定义的:
public interface Setextends Collection { public abstract boolean add(E e); public abstract boolean remove(Object o); ... }
根据上面的定义,add方法中传过去的参数类型一定要与Set中定义的类型一致,否则编译器会报错。但是,remove方法却无此限制,它接受的类型是Object,所以你可以remove任何类型,这就是所谓的type safe。这是在1.5中对Collection的generifying过程中,为了保留对以前版本中方法的支持而造成的。
2 对程序做如下改动就可以解决问题: (short)(i-1); 即把(i-1)的int再cast成short,然后它会被autobox成Short对象,这样前面指示行里在Set中加的Short就会被删除,最后的答案也就变成(a)了。所以,任何牵扯到不同数值类型之间的运算(byte, short, char和int,long),返回的值一定是int或long。 一定要小心,尽量不要混合使用。尽量使用int或long。