Integer大小的比较

今天发现一个挺诡异的问题,先来看看下面这段代码:

public class Test {  
    public static void main(String[] args) {  
        Integer a = 1000;  
        Integer b = 1000;  
        System.out.println("a==b : " + (a == b));  
        System.out.println("a.equals(b) : " + a.equals(b));  
    }  
}  

这段代码的输出结果是这样的:

a==b : false  
a.equals(b) : true  


对于基本类型,以及基本类型的比较,我一直都是用“==”来进行比较的。一直没注意这个问题,今天遇到了吓一大跳。庆幸过去的程序中用“==”没出现问题。把源码拖出来看了一下,明白了,解释如下:

平常我们使用Integer a = xxx;的时候,Integer类实际上是调用的是public static Integer valueOf(int i)方法。这个方法是这样的。

   public static Integer valueOf(int i) {  
final int offset = 128;  
if (i >= -128 && i <= 127) { // must cache   
    return IntegerCache.cache[i + offset];  
}  
       return new Integer(i);  
   }  

从上面代码可以看出,当进行赋值操作时,Java将这个值分为两个区间来处理,即:

i 属于[-128, 127]的时,返回IntegerCache.cache[i + offset];

i 属于上面范围以外时,返回new Integer(i)。

上面实例程序中,赋值超出[-128, 127]返回,所以返回为new Integer(1000),而“==”比较是比较内存地址,那么返回值肯定为FALSE。

至此上面陈述的诡异问题就解开了。

但是Java为什么要这样分区间处理呢,这不是使劲儿把我们往误区里勾引吗?为此我测试了这样一个程序:赋值在[-128, 127]之间时。

public class Test {  
    public static void main(String[] args) {  
        Integer a = 10;  
        Integer b = 10;  
        System.out.println("a==b : " + (a == b));  
        System.out.println("a.equals(b) : " + a.equals(b));  
    }  
}  

 这段代码的输出结果是这样的:

a==b : true  
a.equals(b) : true  

为什么出现这种现象呢?这就要问IntegerCache.cache[i + offset]了,跟踪进去代码如下:

static final Integer cache[] = new Integer[-(-128) + 127 + 1];  


static {  
    for(int i = 0; i < cache.length; i++)  
    cache[i] = new Integer(i - 128);  
}  

也就是说,当赋值在[-128, 127]区间时,Java是从同一个数据中取出同一个对象,即内存地址一样,所以“==”操作返回TRUE;

那么猜测Java这样做的目的,可能[-128, 127]区间比较常用,这个区间内的Integer对象也作为静态变量初始化完成,这样直接返回对象可以提高效率。

为了防止不小心掉入这样的陷阱,对于基本类型的比较,用“==”;而对于基本类型的封装类型和其他对象,应该调用public boolean equals(Object obj)方法(复杂对象需要自己实现equals方法)。



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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值