Integer 和int区别 String和new String()

本文就java语言来讲,探讨一下项目中遇到的坑,以及如何避免这个问题。

首先说一下我遇到的坑,我用Integer进行比较,且比较符号为 =,然后我就debug,找半天最后发现这个问题。

那为什么Integer不能用等于号呢?首先先得说一下装箱和拆箱。

装箱

int 这种基本类型转成Integer对象,可以通过如下代码转换

// 显式转换
int a=7;
Integer aNum = Integer.valueOf(a);
// 隐式转换
Integer b=7;

拆箱

Integer对象转换成int

Integer c=new Integer(6);
int d=c.intValue();

讲完装拆箱,再来说一下new 关键字,new Integer,会直接在堆内存中分配一对象

Integer a=new Integer(5);
Integer b=new Integer(5);

a,b是两个对象,他们在内存中的地址是不一样的。

== 和equals的区别

如果比较的是基本类型的情况下,==,就是比较的内容,而如果是Integer这种对象的情况下,比较的是引用,也就是内存地址,而equal是Object才有的方法,一般都需要手动实现,他比较的是对象的内容

public boolean equals(Object obj) {
    if (obj instanceof Integer) {
        return value == ((Integer)obj).intValue();
    }
    return false;
}

通过上面也说明了一个问题,如果两个Integer进行比较使用==比较的是两个地址,所以会返回false,所以我们需要使用equals,但是<,<=,>,>= 是不受影响的。

那Integer和int比较会出现什么问题呢?首先看几个例子。

Integer a=5;
int b=5;
Integer c=128;
int d=128;
Integer e=128;
Integer f=5;
a==b?  true,因为Integerint比较会转换成int类型
c==d? true,同样会转成int类型进行比较
a==f?  true
c==e?  false 为啥a==f,c!=e呢?看一下源码
private static class IntegerCache {
        static final int low = -128;
        static final int high;
        static final Integer cache[];

        static {
            // high value may be configured by property
            int h = 127;
            String integerCacheHighPropValue =
                sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
            if (integerCacheHighPropValue != null) {
                try {
                    int i = parseInt(integerCacheHighPropValue);
                    i = Math.max(i, 127);
                    // Maximum array size is Integer.MAX_VALUE
                    h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
                } catch( NumberFormatException nfe) {
                    // If the property cannot be parsed into an int, ignore it.
                }
            }
            high = h;

            cache = new Integer[(high - low) + 1];
            int j = low;
            for(int k = 0; k < cache.length; k++)
                cache[k] = new Integer(j++);

            // range [-128, 127] must be interned (JLS7 5.1.7)
            assert IntegerCache.high >= 127;
        }

        private IntegerCache() {}
    }

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

通过源码可以知道,为了更好的空间和时间性能,Integer会缓存频繁使用的数值,数值范围为-128到127,在此范围内直接返回缓存值。IntegerCache.low 是-128,IntegerCache.high是127,如果在这个区间内,他就会把变量i当做一个变量,放到内存中;但如果不在这个范围内,就会去new一个Integer对象,
而如果两个Integer值都不在这个范围内,那么就会new了两个对象实例,两个对象用==比较肯定是false。


String a=new String("abc") ;
String b=new String("abc") ;
String c="abc";
String d="abc";
System.out.println(a==b); false
System.out.println(c==d); true

请添加图片描述

JVM 会把内存存在两个区域,堆和字符串池,一般new 的对象都会放到堆中,而String c="abc"会把c这个对象放到字符串池中。
String t=new String(“abc”),会涉及到几个动作。 采用new关键字新建一个字符串对象时,JVM首先在字符串池中查找有没有"abc"这个字符串对象,如果有,则不在池中再去创建"abc"这个对象了,直接在堆中创建一个"abc"字符串对象,然后将堆中的这个"abc"对象的地址返回赋给引用,这样,a就指向了堆中创建的这个"abc"字符串对象;如果没有,则首先在字符串池中创建一个"abc"字符串对象,然后再在堆中创建一个"abc"字符串对象,然后将堆中这个"abc"字符串对象的地址返回引用,这样,t指向了堆中创建的这个"abc"字符串对象。

采用字面值的方式创建一个字符串时,JVM首先会去字符串池中查找是否存在"abc"这个对象,如果不存在,则在字符串池中创建"abc"这个对象,然后将池中"abc"这个对象的引用地址返回给字符串常量c,这样c会指向池中"abc"这个字符串对象;如果存在,则不创建任何对象,直接将池中"abc"这个对象的地址返回,赋给字符串常量。

一下是Integer和int对比,如果感觉有用,可以点个关注,以后输出更多有用的内容。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值