1.装箱拆箱
Java中有8种基本类型,但鉴于Java面向对象的特点,它们同样有着对应的8个包装类型,比如int和Integer,包装类型的值可以为null(基本类型没有null值,而数据库的表中普遍存在null值。所以实体类中所有属性均应采用封装类型),很多时候,它们都能够相互赋值。
请看如下实例
通过观察字节码,我们发现:
1、赋值操作使用的是Integer.valueOf方法。
2、在进行乘法运算的时候,调用了Integer.intValue方法来获取基本类型的值。
3、在方法返回的时候,再次使用了Integer.valueOf方法对结果进行了包装。 这就是Java中的自动装箱拆箱的底层实现。
2.IntegerCache
这里有一个陷阱问题,我们继续跟踪Integer.valueOf方法
这个IntegerCache,缓存了low和high之间的Integer对象。默认low是-128,high是127
可以通过-XX:AutoBoxCacheMax来修改上限,上限可以变,但是只能大于等于127.
下面是一道经典的面试题,请考虑一下运行代码后,会输出什么结果?
package sandwich.test5;
/**
* @author 公众号:三明治聊IT
* @date 2021/6/12
* -XX:AutoBoxCacheMax=256
*/
public class IntegerCacheTest {
public static void main(String[] args) {
Integer n1 = 123;
Integer n2 = 123;
Integer n3 = 130;
Integer n4 = 130;
System.out.println(n1 == n2);
System.out.println(n3 == n4);
}
}
结果是
true
false
一般情况下是是true,false因为缓存的原因。(在缓存范围内的值,返回的是同一个缓存值,不在的话,每次都是new出来的) 当我加上VM参数 -XX:AutoBoxCacheMax=256执行时,结果是true,ture,扩大缓存范围,第二个为true原因就在于此。