从根上读懂阿里巴巴手册 | 为什么 Integer 不能使用 ==

✋点击“面试交流”加入交流群✋

“置顶公众号”,每天推送面试专题

阿里巴巴开发手册,(四)OOP 规约,第 7 条解释说:

【强制】所有整型包装类对象之间值的比较,全部使用 equals 方法比较。说明:对于 Integer var = ? 在 - 128 至 127 范围内的赋值,Integer 对象是在 IntegerCache.cache 产 生,会复用已有对象,这个区间内的 Integer 值可以直接使用 == 进行判断,但是这个区间之外的所有数据,都会在堆上产生,并不会复用已有对象,这是一个大坑,推荐使用 equals 方法进行判断。

看下下面的代码,你猜结果是什么呢?请分析 30 秒看下结果。

public class IntegerTest {
    public static void main(String[] args) {
        Integer a = 100,
        b = 100,
        c = 200,
        d = 200;
        System.out.println(a == b);
        System.out.println(c == d);
    }
}

输出结果

true
false

是不是有一些惊喜呢?那么我们从源码上分析一下这个问题,只有你需要面试问题的时候也会说,因为缓存了 -128 到 127 之间的数值,但是为什么就缓存这么一小段呢?会不会其他包装类型也有类似的问题呢?

public static Integer valueOf(int i) {
        if (i >= IntegerCache.low && i <= IntegerCache.high)
            return IntegerCache.cache[i + (-IntegerCache.low)];
        return new Integer(i);
    }

通过源码我们可以看出(Integer.class:892),如果 i 的内容大于 -128,小于 127,就会在 IntegerCache.cache 直接直接获取,所以如果不是在这个区间范围里面就是新对象,== 当然不相同了。我们继续阅读源码可以发现,这个配置是为了缓存对象,提高访问速度。当然我们可以通过-XX:AutoBoxCacheMax=<size>参数设置缓存的上限,也就是 IntegerCache.high。所以是不是灵光一现?如果我们在平时处理一些常数级别的 Integer 的时候,恰好大于 127,那么就可以通过这个参数来提高程序的性能啦。

修改之后我们再看输出结果?

true
true

好的,那么问题又来了,我们看的源码是 valueOf 啊,莫非每次自动装箱的时候都调用这个方法吗?答案是对的,如果你还是想深入研究下,那么可以使用 IDEA 集成一下 javap 工具,反编译一下汇编代码,具体操作如下。我们需要Settings->Tools->External Tools 添加一个扩展工具,详细配置如下 很需要注意的是第二张图

  • Program 需要指定你 JDK 目录中 javap 的位置

  • Arguments 是扩展工具带的参与, -c $FileNameWithoutExtension$.class,所以需要配置这个样子,可以点击 右侧是 Insert Macro… 查看所有关联项

  • Working directory 要配置到 class 的输出目录,包括项目目录和 package 目录,所以内容是 $OutputPath$/$FileDirRelativeToSourcepath$

接下来我们就看到了如下内容

/Library/Java/JavaVirtualMachines/jdk1.8.0_101.jdk/Contents/Home/bin/javap -c IntegerTest.class
Compiled from "IntegerTest.java"
public class com.github.codedrinker.basic.IntegerTest {
  public com.github.codedrinker.basic.IntegerTest();
    Code:
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return

  public static void main(java.lang.String[]);
    Code:
       0: bipush        100
       2: invokestatic  #2                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
       5: astore_1
       6: bipush        100
       8: invokestatic  #2                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
      11: astore_2
      12: sipush        200
      15: invokestatic  #2                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
      18: astore_3
      19: sipush        200
      22: invokestatic  #2                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
      25: astore        4
      27: getstatic     #3                  // Field java/lang/System.out:Ljava/io/PrintStream;
      30: aload_1
      31: aload_2
      32: if_acmpne     39
      35: iconst_1
      36: goto          40
      39: iconst_0
      40: invokevirtual #4                  // Method java/io/PrintStream.println:(Z)V
      43: getstatic     #3                  // Field java/lang/System.out:Ljava/io/PrintStream;
      46: aload_3
      47: aload         4
      49: if_acmpne     56
      52: iconst_1
      53: goto          57
      56: iconst_0
      57: invokevirtual #4                  // Method java/io/PrintStream.println:(Z)V
      60: return
}

我们可以清楚的看到 2,8,15,22 行对应内容是 Integer.valueOf,所以由此可以确定,Java 在编译代码的时候,会把 Integer a = 转换为 valueOf 来赋值,是不是到此所有问题都迎刃而解?

对了,差点忘记刚才的问题,是不是别的封装类也有这个问题呢?我们继续阅读源码发现 CharacterLongShort 是不是同样有这个问题呢?好的,那么反编译 Long 的任务就交给你了。

1. 全栈架构之打包推荐【建议收藏,常读】

2. 一个空格引发的“惨案“

3. 分布式系统中Session共享的常用方案

4Java语言“坑爹”排行榜TOP 10

5. 我是一个Java类(附带精彩吐槽)

6. mysql索引失效,差点我的工作凉了

7. 既生synchronized,何生volatile?

8. 微服务一直火,为什么服务化要搞懂?

9. MySQL的COUNT语句,不简单!

10. 漫画:HashSet和TreeSet实现与原理

扫码二维码关注我

·end·

—如果本文有帮助,请分享到朋友圈吧—

我们一起愉快的玩耍!

你点的每个赞,我都认真当成了喜欢

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值