面试题

说下面的整数类型之间比较的结果,以及原因

  1. Integer i = new Integer(50);
  2. Integer j = new Integer(50);
  3. System.out.print(i == j); // ?
  4. Integer i = new Integer(50);
  5. Integer j = 50;
  6. System.out.print(i == j); // ?
  7. Integer i = 50;
  8. Integer j = 50;
  9. System.out.print(i == j); //?
  10. Integer i = 128;
  11. Integer j = 128;
  12. System.out.print(i == j); //?
  13. Integer i = new Integer(50);
  14. int j = 50
  15. System.out.print(i == j); //?
  16. Integer i = 50;
  17. int j = 50
  18. System.out.print(i == j); //?

大家都知道答案吗?

分析

主要考察你对基础知识的掌握,基本类型和包装类型,以及包装类型的缓存机制。

答案

最终输出的结果如下:

  1. Integer i = new Integer(50);
  2. Integer j = new Integer(50);
  3. System.out.print(i == j); // false
  4. Integer i = new Integer(50);
  5. Integer j = 50;
  6. System.out.print(i == j); // false
  7. Integer i = 50;
  8. Integer j = 50;
  9. System.out.print(i == j); // true
  10. Integer i = 128;
  11. Integer j = 128;
  12. System.out.print(i == j); // false
  13. Integer i = new Integer(50);
  14. int j = 50
  15. System.out.print(i == j); // true
  16. Integer i = 50;
  17. int j = 50
  18. System.out.print(i == j); // true

首先我们区分下几个概念:

  1. Integer 是int的包装类, int是java中的基本类型,他们可以进行自动装箱、拆箱
  2. Integer的默认值是null, int的默认值是0
  3. Integer是一个对象,数据存储在堆山,变量存储着堆中的地址引用。int直接存储数据值,加载到栈上运算。

    场景一: 两个new Integer()变量比较

    结论: 两个new Integer()变量比较,永远false
    原因:因为new Integer()相当于在堆中开辟了一个新的空间,存放数据,内存地址必然不通,所以为false。
    1. Integer i = new Integer(50);
    2. Integer j = new Integer(50);
    3. System.out.print(i == j); // false

    场景二: Integer和new Integer()变量比较

    结论: Integer变量和new Integer()变量比较,永远false
    原因:new Integer()指向的是堆中新建对象的地址, Integer x = 1中指向的可能堆中的地址也有可能是缓存中的数据,无论哪种情况,他们都是不一致的。
    1. Integer i = new Integer(50);
    2. Integer j = 50;
    3. System.out.print(i == j); // false

    场景二: 两个Integer变量比较

    ```java Integer i = 50; Integer j = 50; System.out.print(i == j); // true

Integer i = 128; Integer j = 128; System.out.print(i == j); // false

  1. **结论:** 两个`Integer`变量比较,如果变量值在区间-128127之间,则结果为true, 如果变量不在次范围内,结果为false。<br />**原因:**`Integer i = 50`, 编译后会自动装箱变成 `Integer i = Integer.valueOf(50)`, 然后我们看下valueOf的源码如下:
  2. ```java
  3. public static Integer valueOf(int i) {
  4. // 如果i大于等于IntegerCache中的-128,小于等于127,就会从缓存中获取,
  5. // 而缓存中的caches在初始化new出来
  6. if (i >= IntegerCache.low && i <= IntegerCache.high)
  7. return IntegerCache.cache[i + (-IntegerCache.low)];
  8. // 如果不在上面的区间内,则新建对象
  9. return new Integer(i);
  10. }

image.png
根据源码得知,如果是在缓存范围内,会直接从缓存中获取,那么他们取到的是同一个对象,他们的地址也是一样的, == 为true, 否则为false。

场景二: 基本类型和Integer、new Integer()变量比较

  1. Integer i = new Integer(50); //自动拆箱为 int i=100; 此时,相当于两个int的比较
  2. int j = 50
  3. System.out.print(i == j); // true
  4. Integer i = 50;
  5. int j = 50
  6. System.out.print(i == j); // true

结论: 基本类型int和Integer、new Integer()变量比较时,只要两个值相等,则为true
原因:包装类Integer遇到基本类型int时,自动会拆箱成int, 实际上就变成了两个int变量的比较。我们查看编译后的结果:i.intValue() == j,通过调用intValue()方法进行拆箱操作。

  1. public int intValue() {
  2. return value;
  3. }

衍生扩展

因为 == 比较的不可靠,所以建议大家进行等值比较的时候用equals, 上面无论哪种情况,用equals准没错。
不仅int,Java中的另外7中基本类型都可以自动装箱和自动拆箱,其中也有用到缓存。见下表:
image.png