需要注意自动装拆箱的一个特例
来源:互联网 发布:js获取表格一行数据 编辑:程序博客网 时间:2024/04/29 21:48
首先看一段代码(使用JDK 5),如下:
public class Test {
public static void main(String[] args) {
Integer i1 = 127;
Integer i2 = 127;
public static void main(String[] args) {
Integer i1 = 127;
Integer i2 = 127;
if (i1 == i2)
System.out.println("Equal!");
else
System.out.println("Not equal!");
}
}
System.out.println("Equal!");
else
System.out.println("Not equal!");
}
}
输出结果想必大家也知道,是“Equal!”。现在把i1和i2的值由127改为128看看会发生什么?结果输出“Not equal!”。
为了方便说明问题,我写了下面的代码:
public class Test {
public static void main(String[] args) {
Integer i1 = 127;
Integer i2 = 127;
Integer i3 = Integer.valueOf(127);
public static void main(String[] args) {
Integer i1 = 127;
Integer i2 = 127;
Integer i3 = Integer.valueOf(127);
if (i1 == i2)
System.out.println("i1 == i2 is true!");
else
System.out.println("i1 == i2 is false!");
System.out.println("i1 == i2 is true!");
else
System.out.println("i1 == i2 is false!");
if (i1 >= i2)
System.out.println("i1 >= i2 is true!");
else
System.out.println("i1 >= i2 is false!");
System.out.println("i1 >= i2 is true!");
else
System.out.println("i1 >= i2 is false!");
if (i1 == i3)
System.out.println("i1 == i3 is true!");
else
System.out.println("i1 == i3 is false!");
System.out.println("i1 == i3 is true!");
else
System.out.println("i1 == i3 is false!");
}
}
}
当值是127时,输出是:
i1 == i2 is true!
i1 >= i2 is true!
i1 == i3 is true!
i1 >= i2 is true!
i1 == i3 is true!
当值是128时,输出是:
i1 == i2 is false!
i1 >= i2 is true!
i1 == i3 is false!
i1 >= i2 is true!
i1 == i3 is false!
说明:
我使用的是Sun JDK 1.5.0_03-b07 和 Eclipse 3.2M4。
“Integer i1 = 127;”在JDK1.4下不能编译通过的,会提示:“ Type mismatch: cannot convert from int to Integer”的错误,一般改写为:“Integer i1 = new Integer(127);”。
“Integer i1 = 127;”在JDK1.5下可以编译通过的,这就是自动装箱(Autoboxing)和自动拆箱(Auto-Unboxing)。自动装箱(Autoboxing)特性让Java自动包装一个简单数据类型(例如int)到对应的包装类型中(例如Integer)中。
在《JSR 201: Extending the Java Programming Language with Enumerations, Autoboxing, Enhanced for loops and Static Import》中,对这个问题,是作了这样的规定:
If the value p being boxed is true, false, a byte, an ASCII character, or an integer or short number between -127 and 128, then let r1 and r2 be the results of any two boxing conversions of p. It is always the case that r1 == r2.
在Java中,The following is the list of primitives stored as immutable objects(不可变对象):
* boolean values true and false
* All byte values
* short values between -128 and 127
* int values between -128 and 127
* char in the range /u0000 to /u007F
* All byte values
* short values between -128 and 127
* int values between -128 and 127
* char in the range /u0000 to /u007F
为了更容易理解问题,用Jad将上面代码反编译,如下:
import java.io.PrintStream;
public class Test
{
{
public Test()
{
}
{
}
public static void main(String args[])
{
Integer i1 = Integer.valueOf(128);
Integer i2 = Integer.valueOf(128);
Integer i3 = Integer.valueOf(128);
{
Integer i1 = Integer.valueOf(128);
Integer i2 = Integer.valueOf(128);
Integer i3 = Integer.valueOf(128);
if(i1 == i2)
System.out.println("i1 == i2 is true!");
else
System.out.println("i1 == i2 is false!");
if(i1.intValue() >= i2.intValue())
System.out.println("i1 >= i2 is true!");
else
System.out.println("i1 >= i2 is false!");
if(i1 == i3)
System.out.println("i1 == i3 is true!");
else
System.out.println("i1 == i3 is false!");
}
}
从这可以看出,“Integer i1 = 127;”在JDK1.5下应该编译成了“Integer i1 = Integer.valueOf(127);”。
再看看java.lang.Integer中关于valueOf的源代码是怎样的:
public static Integer valueOf(int i) {
final int offset = 128;
if (i >= -128 && i <= 127) { // must cache
return IntegerCache.cache[i + offset];
}
return new Integer(i);
}
final int offset = 128;
if (i >= -128 && i <= 127) { // must cache
return IntegerCache.cache[i + offset];
}
return new Integer(i);
}
可以看出,这个值在-128到127之间,会将其cached(缓存)起来,如果多次使用的话,会节省内存和改善性能;如果不在这个范围之内,则生成一个新的Integer Object instance,这样如果进行“==”时,由于是比较两个不同的Object references,故结果是false。事实上,这个特性从JDK 1.3就存在了(以前的我不清楚) 。
确切的说,上面出现的问题实质并不是Autoboxing和Auto-Unboxing,应该是Integer类的valueOf(int i)方法,只是Autoboxing和Auto-Unboxing将这个问题掩盖了。
- 需要注意自动装拆箱的一个特例
- 需要注意自动装拆箱的一个特例
- 需要注意自动装拆箱的一个特例
- 自动装拆箱的一个特例
- c++的一些特例,学习的时候需要注意!
- 宽带691的一个特例
- 文件空间变大的一个特例
- C++中重载的一个特例
- 一个动态分配二维数组程序的特例
- 泛型编程的一个特例
- SWT绘图需要注意的一个问题
- “多态”一个需要注意的问题
- JS需要注意的一个问题
- 集合初始化需要注意的一个问题
- oracle 默认值需要注意的一个地方
- 赋权时需要注意的一个现象
- 使用synchronized需要注意的一个问题
- 断言(Assertion)需要注意的一个地方
- J2ME(CLDC/MIDP)简介
- 移动无title窗体
- DTD - 元素
- 回学校三天了
- 第一天
- 需要注意自动装拆箱的一个特例
- WINDOWS系统文件详解
- Oracle 11g new feature1
- 移动SI之MAS培训后一些总结
- CrystallReportViewer控件类常用属性与方法
- DTD - 属性
- 我爱娃娃
- 2007年步入社会的第一年,开始实习的第二天,留下我的第一篇Blog
- DTD - 实体