Q1 为什么要为每个基本数据类型设计封装类?
- Java本身是一门面向对象语言,具有“一切皆对象”的特点,对象是Java的基础操作单元。
- 很多时候传递数据需要引用类型,举个例子,比如List、Set这些集合类都只能存储引用类型。
- 封装类型还有很多好处,例如安全性较好。
- 具体来说,Integer内部维护了一个字段
private final int value;
,可见这是一个不可变的基本类型变量。也就是说,一个Integer对象一旦被创建出来,其代表的值也就不会再改变了,我们可以说Integer对象是不可变的。这种不可变性有助于编写高效的、线程安全的程序,因为不可变对象不需要担心并发修改的问题。
- 具体来说,Integer内部维护了一个字段
Q2 Integer和int有什么区别?
- Integer是一个类,int是基本数据类型。
- Integer的默认值是null,int的默认值是0。
- Integer对象被存储在堆内存中,作为局部变量的int变量存储在栈空间中。
- 作为成员变量的基本数据类型会被存放在堆内存中,具体来说,它会被放在对象中的”实例数据“部分中。而方法中的局部变量会被存放在栈帧中的局部变量表中。
TIP 这里列一个表格,解释一下各种变量究竟存放在内存中的什么区域。
成员变量 | 成员变量是被定义在类中的字段(Filed),是实例对象的一部分,随着对象的创建被存放在堆中。 |
---|---|
静态变量 | 同样被定义在类中,但被static关键字修饰。是类的一部分,JDK 7以前被存放在方法区中,在JDK 7及以后被转移到堆区中。总之,JDK7以后,随着类的加载,静态变量被存放到堆区中。 |
局部变量 | 某个方法被调用时,其中的局部变量被放入虚拟机栈的栈帧中的局部变量表中,方法执行结束后栈帧从虚拟机栈中弹出,所以局部变量存放在虚拟机栈中。 |
Q3 解释一下自动拆箱、自动装箱?缓存池机制知道吗?
- 以Integer类举例,自动拆箱与自动装箱实际上涉及了Integer中的两个方法。
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
public int intValue() {
return value;
}
3个例子,描述Java中发生自动装箱的3种情况。
// 1.将基本数据类型赋值给对应的包装类类型的变量时,
// 实际上调用了Integer valueOf(int)方法,发生自动装箱。
int i = 10;
Integer obj = i;
// 2.将基本数据类型作为参数传递给需要封装类类型的方法时。
public void printInteger(Integer obj) {
System.out.println(obj);
}
int i = 10;
printInteger(i); // 自动装箱,将 int 类型的值 10 转换为 Integer 对象
// 3.数学运算或者比较操作
Integer a = 10;
Integer b = 20;
// 下面这行代码实际上同时发生了自动拆箱和自动装箱
// 首先,a、b两个Integer对象会先调用int intValue()来拆箱
// 拆箱后的两个int变量进行相加,然后再调用Integer valueOf(int)装箱成对象c
// 为什么会这样呢?因为在Java中,加号运算符的操作数只能是基本数据类型或者字符串
Integer c = a + b;
3个例子,描述Java中发生自动拆箱的3种情况。
// 1.赋值
Integer obj = 10;
int i = obj;
// 2.方法传参
public void printInt(int i) {
System.out.println(i);
}
Integer obj = 10;
printInt(obj); // 调用int intValue()方法自动拆箱
// 3.数学运算或者比较操作
Integer a = 10;
Integer b = 20;
int c = a + b;
- 封装类的缓存池机制实际上涉及到自动装箱中的Integer valueOf(int i)方法,只有Character、Byte、Short、Integer、Long有缓存池,其中Character的范围为0~127, Byte、Short、Integer、Long的范围为-128~127。
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
// 这里的cache实际上就是Integer的缓存池,保存了-128~127之间的Integer对象
// 这个cache实际上是Integer的私有静态内部类private static class IntegerCache
// 的静态成员static final Integer[] cache;
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}