Java谜题选 - I2011-01-29 yeeyan 译: 易晓斓题目1:Sets的乐趣程序public class ShortSet{
public static void main(String args[]) {
Set<Short> s = new HashSet<Short>();
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 Set<E>extends Collection<E> {
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。