含义
自动装箱:自动将基本数据类型转换为包装器类型
自动拆箱:自动将包装器类型转换为基本数据类型
//自动装箱
Integer total = 99;
//自动拆箱
int totalprim = total;
- 上述执行的第一条Integer total=99,事实上执行的是total=Integer.valueOf(99)
- 而第二条totalprim=99事实上是执行的total.intValue()转换
Integer的value函数源码如下:
public static Integer valueOf(int i) {
return i>=128 || i<-128? new Integer[i]:SMALL_VALUES[i+128];
}
- 上述源码表明如果i在[-128,127)区间内,会执行一个SMALL_VALUES,这个是一个静态的Integer数组对象,当取值在这个范围内会自动返回已经创建好的对象而不是重新生成。不在这个范围内的就会重新new一个对象。
装箱拆箱相关问题
- 对于Integer类型因为创建过程中有一个比较创建的过程,那么返回的对象之间就会有一些关系,如下代码中所示。i1和i2都是100的时候就会直接从SMALL_VALUES中返回同一个对象,自然就相等,当超过上述区间范围的时候就会重新new一个对象,i3和i4指向两个对象自然不相等
public class Main {
public static void main(String[] args) {
Integer i1 = 100;
Integer i2 = 100;
Integer i3 = 200;
Integer i4 = 200;
System.out.println(i1==i2);//true
System.out.println(i3==i4);//false
}
}
- Integer派别对象表:
类型 | 相同对象范围 | 不同对象范围 | ||
---|---|---|---|---|
Integer | [-128, 127) | i>=128 | i<-128 | |
Short | [-128, 127) | i>=128 | i<-128 | |
Character | c<128 | c>=128 | ||
Long | [-128, 127) | i>=128 | i<-128 |
对于Double和Float类型每一次都是重新new一个对象
- 再根据以上情况深入理解
//进行自动拆箱操作
Integer num1 = 400;
int num2 = 400;
System.out.println(num1==num2);//true
//进行内容比较
System.out.println(num1.equals(num2));//true
- equal源码(Integer的),首先两者类型要相同然后比较其中的内容(重写的,默认的Object中的equal方法是比较地址)
public boolean equals(Object o) {
return (o instance of Integer) && ((Integer)o).value == value;
}
- 当一个数据类型与封装类型进行==、+、-、*、/运算时,会将封装类进行拆箱,对基础数据类型进行运算。如下中所示,当num3==(num1+num2)比较的时候就会进行拆箱最后比较的是int数值自然相等。而equal方法在比较内容前要先比较对象的类型,因为Long和Integer类型不同所以返回false
Integer num1 = 100;
int num2 = 100;
Long num3 = 200;
System.out.println(num3==(num1+num2));//true
System.out.println(num3.equals(num1+num2));//false
- ==两边都是包装类型的时候,如果有运算符则会进行拆箱操作进行比较,否则就是比较对象地址。如下所示,对于两个Integer的类型相加运算此时的比较会进行拆箱,最后比较的是300和300两个int数值自然就相等。
Integer num1 = 100;
Integer num2 = 200;
Long num3 = 300;
System.out.println(num3==(num1+num2));//true
网友评论