面试官:兄弟,说说基本类型和包装类型的区别吧 (qq.com)

面试官:你知道包装类的缓存机制吗? (qq.com)

1、包装类型可以为null,基本类型不中

这点使得包装类型可以应用于POJO中,而基本类型则不行。

POJO:Plain Ordinary Java Object,简单无规则的 Java 对象,只有属性字段以及 setter 和 getter 方法。

和 POJO 类似的,还有数据传输对象 DTO(Data Transfer Object,泛指用于展示层与服务层之间的数据传输对象)、视图对象 VO(View Object,把某个页面的数据封装起来)、持久化对象 PO(Persistant Object,可以看成是与数据库中的表映射的 Java 对象)。

那为什么 POJO 的属性必须要用包装类型呢?

数据库的查询结果可能是 null,如果使用基本类型的话,因为要自动拆箱(将包装类型转为基本类型,比如说把 Integer 对象转换成 int 值),就会抛出 NullPointerException 的异常。

2、包装类型可用于泛型,基本类型不中

泛型不能使用基本类型,因为使用基本类型时会编译出错。

List<int> list = new ArrayList<>(); // 提示 Syntax error, insert "Dimensions" to complete ReferenceType
List<Integer> list = new ArrayList<>();

为什么呢?因为泛型在编译时会进行类型擦除,最后只保留原始类型,而原始类型只能是 Object 类及其子类——基本类型是个特例。

3、基本类型比包装类型更高效

基本类型在栈用直接存储的具体数值,包装类型则存储的是堆中的引用。

alt

相对于基本类型而言,包装类型占用更多的内存空间。

4、存储位置

基本数据类型的局部变量存放在Java虚拟机栈的局部变量表中,基本数据类型的成员变量(未被static修饰)存放在Java虚拟机的堆中。

包装类型属于对象类型,几乎所有的对象事例都存储在堆中。

追问:那静态成员变量(static修饰)在JVM的哪里?

static 关键字详解 (qq.com)

被static修饰的成员属于类,被类中所有对象共享。

被static 声明的成员变量属于静态成员变量,静态变量 存放在 Java 内存区域的方法区。

追问:自动装箱和自动拆箱了解吗?

既然有了基本类型和包装类型,肯定有些时候要在它们之间进行转换。

把基本类型转换成包装类型的过程叫做装箱(boxing)。

反之,把包装类型转换成基本类型的过程叫做拆箱(unboxing)。

在 Java SE5 之前,开发人员要手动进行装拆箱,比如说

Integer chenmo = new Integer(10);  // 手动装箱
int wanger = chenmo.intValue();    // 手动拆箱

Java SE5 为了减少开发人员的工作,提供了自动装箱与自动拆箱的功能。

Integer chenmo  = 10;  // 自动装箱
int wanger = chenmo;   // 自动拆箱

追问:包装类型的缓存机制了解吗?

在JDK5中引入了包装类的缓存机制,有助于节省内存。

Byte、Short、Integer、Long这四种包装类型默认创建了【-128,127】的相应类型的缓存数据;

Character创建了数值在【0,127】范围的缓存数据;

Boolean直接返回True or False

浮点数类型的包装类Float,Double并没有实现常量池技术。(😁太多了缓不过来)

追问:什么时候会使用缓存机制,什么时候不会呢?

使用构造函数创建对象时不使用缓存。例如:Integer a = new Integer(123);

使用整型包装类时,使用valueOf()方法,会从缓存中取值,如果命中缓存,会减少资源的开销,parseXXX方法没有这个机制。

追问:下题输出什么

// 1)基本类型和包装类型
int a = 100;
Integer b = 100;
System.out.println(a == b);

// 2)两个包装类型
Integer c = 100;
Integer d = 100;
System.out.println(c == d);

// 3)自动装箱
c = 200;
d = 200;
System.out.println(c == d);

提示:

==是判断两个变量或实例是不是指向同一个内存空间。

equals是判断两个变量或实例所指向的内存空间的值是不是相同。

答案:

第一段代码,基本类型与包装类型进行==比较,这时候b会自动拆箱,直接和a比较值,所以结果为true;

第二段代码,两个包装类型都被赋值100,会进行自动装箱,但是存在缓存,因此为true;

第三段代码,两个包装类型都被赋值200,会进行自动装箱,但是超出缓存范围【-128,127】,因此为false;