【小家java】大杂烩---那些年我们一起躺过的坑

1、概述

此篇博文没有具体的主题,主要针对于平时开发过程中遇到的一些小问题的记录,并且大都从源码的角度去解释为什么会报错。并且此篇博文是持续更新中。。。

2、栗子

1、包装类型的的> < =的比较

public static void main(String[] args) {
    Integer i = null;
    Integer i2 = null;
    System.out.println(i < i2); //java.lang.NullPointerException
}

包装类型能用比较运算符的根本原因:JDK5的自动拆箱。因此如果是是null的话,自动拆箱就报错了。所以在日常使用的时候一定要注意。特别是Interger和int比的时候,如果Interger为null,那必然空指针。从代码角度看是因为自动拆箱JVM会调用Interger的.intValue()方法,所以如果是null,就挂了

2、关于int值比较的一些问题
在日常编码中,经常会遇到Interger的比较问题(比如id相等),看下面例子

 public static void main(String[] args) {
   Integer i1 = 100;
   Integer i2 = 100;
   System.out.println(i1 == i2); //true
   Integer i3 = 200;
   Integer i4 = 200;
   System.out.println(i3 == i4); //false
 }

我们知道,对象比较用==的话比较的是地址值,所以咱们看看两者的地址值哈。

public static void main(String[] args) {
    Integer i1 = 100;
    Integer i2 = 100;
    System.out.println(System.identityHashCode(i1)); //2093631819
    System.out.println(System.identityHashCode(i2)); //2093631819
    Integer i3 = 200;
    Integer i4 = 200;
    System.out.println(System.identityHashCode(i3)); //2074407503
    System.out.println(System.identityHashCode(i4)); //999966131
}

各位看官,应该看出端倪了吧,这就是为什么第一个是true,第二个为false的最直接原因,但还不是最根本原因,下面容我介绍一下最根本原因,Interger内部有如下代码:
这里写图片描述
同理Long里面一样
这里写图片描述
如图可以看出,Interger在初始化的时候内部就维护了一个缓存:-128到127对象都给Cache了,所以当我们向上转型这些对象时,没有new而是从缓存拿的,所以就是上面的答案。当然,我们必须清楚,其实JVM自动装箱调用了valueOf方法:

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

我们发现,没有new而是从缓存拿,因此内存地址值肯定一样的。所以可以很好理解下面这两个现象了:

 public static void main(String[] args) {
     Integer i1 = new Integer(100);
     Integer i2 = new Integer(100);
     System.out.println(i1 == i2); //false 因为用了new关键字,一定开辟新内存的
     Integer i3 = new Integer(200);
     Integer i4 = new Integer(200);
     System.out.println(i3 == i4); //false
 }

这样比较也没有问题:

public static void main(String[] args) {
    int i1 = 100;
    int i2 = 100;
    System.out.println(i1 == i2); //true
    int i3 = 200;
    int i4 = 200;
    System.out.println(i3 == i4); //true
}

所以,综上所述,同类型的包装类型比较的时候,建议使用equals()方法。但是此处提醒一点:不同类型的equals肯定是false的,比如Interger和Long的对比,不要这么来:

 public static void main(String[] args) {
     Integer i1 = 100;
     Long i2 = 100L;
     System.out.println(i1.equals(i2)); //false
 }

这个虽然值一样,但结果肯定false。这里我贴一处源码,大家就能了解了:
这里写图片描述
我们发现他们都重写了equals方法,而类型相同是第一必须。so,以后比较的时候一定要注意类型一致啊

3、intValue(),Integer.valueOf(String s)和Integer.parseInt(String s)有什么区别?
intValue()是把Integer对象类型变成int的基础数据类型; //一般由jvm自己调用
静态方法parseInt()是把String 变成int的基础数据类型
静态方法valueOf()是把给定的String参数转化成Integer对象类型,依赖于parseInt()方法;

然后如果想得到一个小的Integer对象,但是Integer.valueOf的效率比new的效率高,因为valueOf有缓存。

综上所述,一般用Integer.parseInt(str),除非你要返回Integer类型,不然还有封装拆箱,性能多少会耗费些。

3、使用场景

持续更新中。。。

4、最后

整理出来的都是一些小细节,希望能帮助到大家

—-题后语—-

我的微信公众号也会持续推送技术干货,欢迎下方二维码扫码关注获取。
这里写图片描述
更多内容持续更新中,欢迎关注我的博客!
有任何问题,可以跟我留言讨论,欢迎指正,不胜感激。
有任何疑问,亦可扫码向我提我哟~

猜你喜欢

转载自blog.csdn.net/f641385712/article/details/80443741