1、 int 和 Integer 区别

1)Integer是int的包装类,int则是java的一种基本数据类型

2)Integer变量必须实例化后才能使用,而int变量不需要

3)Integer实际是对象的引用,当new一个Integer时,实际上是生成一个指针指向此对象;而int则是直接存储数据值

4)Integer的默认值是null(对象的引用,默认为null),int的默认值是0

 

2、 == 比较(同时比较 值和地址)

2.1、Integer变量实际上是对一个Integer对象的引用,所以两个通过new生成的Integer变量永远是不相等的,因为new生成的是两个对象,其内存地址不同,双等号 == 是同时比较值和内存地址的(实际认为比较内存地址即可,因为内存地址相同,值也会相同)

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

 

2.2、Integer变量和int变量比较时,只要两个变量的值是相等的,则结果为true

因为包装类Integer和基本数据类型int比较时,java会自动拆包装Integer为int,然后进行比较,实际上就变为两个int变量的比较

Integer i = new Integer(100);
int j = 100;
System.out.print(i == j); //true

 

2.3、非new生成的Integer变量和new Integer()生成的变量比较时,结果为false。

因为非new生成的Integer变量指向的是java常量池中的对象,

而new Integer()生成的变量指向堆中新建的对象,

两者在内存中的地址不同

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

 

2.4、对于两个非new生成的Integer对象,进行比较时,如果两个变量的值在区间-128到127之间,则比较结果为true,如果两个变量的值不在此区间,则比较结果为false

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

对于第4条的原因:

java在编译Integer i = 100 ; 时,会翻译成为 Integer i = Integer.valueOf(100);

而java API中对Integer类型的valueOf的定义如下:

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

java对于-128到127之间的数,会进行缓存,

Integer i = 127时,会将127进行缓存,下次再写Integer j = 127时,就会直接从缓存中取,就不会new了

 

3、延伸

3.1、理解自动装箱、拆箱

自动装箱与拆箱实际上算是一种“语法糖”。

所谓语法糖,可简单理解为Java平台为我们自动进行了一些转换,保证不同的写法在运行时等价。

因此,它们是发生在编译阶段的,也就是说生成的字节码(bytecode)是一致的。

对于整数,javac替我们自动把装箱转换为Integer.valueOf(),把拆箱替换为Integer.intValue()。可以通过将代码编译后,再反编译加以证实。

原则上,建议避免无意中的装箱、拆箱行为,尤其是在性能敏感的场合,创建10万个Java对象和10万个整数的开销可不是一个数量级的。当然请注意,只有确定你现在所处的场合是性能敏感的,才需要考虑上述问题。毕竟大多数的代码还是以开发效率为优先的。

顺带说一下,在32位环境下,Integer对象占用内存16字节;在64位环境下则更大。

 

3.2、值缓存

就像上一讲谈到的String,Java也为Integer提供了值缓存。

Integer i1 = 1;
Integer i2 = Integer.valueOf(2);
Integer i3 = new Integer(3);

上述代码中第一行与第二行的写法取值使用了值缓存,而第三行的写法则没有利用值缓存。

结合刚刚讲到的自动装箱、拆箱的知识,第一行代码用到的自动装箱,等价于调用了Integer.valueOf(1)。

不仅仅是Integer,Java也为其它包装类提供了值缓存机制,包括Boolean、Byte、Short和Character等。

但与String不同的是,默认都只会将绝对值较小的值放入缓存。

以Integer为例,默认情况下只会缓存-128到127之间的值。当然如果你愿意也可以通过以下JVM参数进行设置:

-XX:AutoBoxCacheMax=N

 

3.3、原始类型操作线程安全吗?

这个问题的正确答案是原始类型操作是“线程不安全,是否有些出乎你的意料?

原始数据类型的变量,需要使用并发相关手段才能保证线程安全。特别的是,部分比较宽的数据类型,比如long、float、double,甚至不能保证更新操作的原子性,可能出现程序读取到只更新了一半数据位的数值!

关于这个话题会在这个专栏后面的并发主题详细介绍。

如果有线程安全的计算需要,建议考虑使用类似 AtomicInteger、AtomicLong 这样线程安全的类。

 

int 转成 Integer

这个在转换的过程中会调用Integer的静态方法valueOf()方法。

源码如下:

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

这里有个IntegerCache其实就是在VM启动时,为了加快int类型到Integer的转化速度,在VM启动时就new出一些int类型与Integer的对应关系,方便直接返回Integer,而不用new,new其实是很耗时的。

注意

这里的 valueOf() 是static 静态方法。因为这时你想要的是Integer对象,所以只能是static的。

 

Integer 到 int 的转化

这个需要调用Integer的方法 intValue(),这个方法不是静态的。

源码如下:

/**
* Returns the value of this {@code Integer} as an
* {@code int}.
*/

public int intValue() {
    return value;
}

这样,当Integer转成int时,如果Integer是null的,则可能会抛空指针异常NullPointerException

Integer integer = 1;

int i = integer;

首先会调用Integer的静态方法valueOf(1),然后调用integer对象的intValue方法。

这时,如果integer对象是null的,在integer.intValue()时,会出现NullPointerException。

javac Test.java
javap -verbose Test

public class Test {
    public static void main(String[] args) {
        Integer integer = 1;
        int i = integer;
        System.out.println(i);
    }
}

 

1)看先反编译代码

public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=3, args_size=1
         0: iconst_1
         1: invokestatic  #2                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
         4: astore_1
         5: aload_1
         6: invokevirtual #3                  // Method java/lang/Integer.intValue:()I
         9: istore_2
        10: getstatic     #4                  // Field java/lang/System.out:Ljava/io/PrintStream;
        13: iload_2
        14: invokevirtual #5                  // Method java/io/PrintStream.println:(I)V
        17: return

一个是 invokestatic、一个是 invokevirtual。

如果Integer integer = null; 的话:

public class Test {
    public static void main(String[] args) {
        Integer integer = null;
        int i = integer;
        System.out.println(i);
    }
}

反编译后:

public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=3, args_size=1
         0: aconst_null
         1: astore_1
         2: aload_1
         3: invokevirtual #2                  // Method java/lang/Integer.intValue:()I
         6: istore_2
         7: getstatic     #3                  // Field java/lang/System.out:Ljava/io/PrintStream;
        10: iload_2
        11: invokevirtual #4                  // Method java/io/PrintStream.println:(I)V
        14: return

解释说明:

intValue() 是把 Integer对象类型变成int 的基础数据类型;

parseInt() 是把 String 变成 int 的基础数据类型;

Valueof() 是把给定的 String参数转化成Integer对象类型;(现在JDK版本支持自动装箱拆箱了)

注意:

intValue() 用法与另外两个不同,比如

int i = new Integer("123"), j = i.intValue(); 相当于强制类型转换(强制类型转换事实上就是调用的这个方法)。

另外两个用法:

Integer.Valueof() , Integer.parseInt() 用的是Interger类名。

i.intValue()用的是对象 i (i 是 Integer 对象)

另外,

Integer a=new Integer(1)

Integer a=Integer.valueOf(1);

两个都是得到一个Integer对象,但是Integer.valueOf的效率高,不需要new操作。

 

 

参考推荐:

Java 中 string 和 int 互相转化

Java基本数据类型sizeof功能

Java中equals和== 区别

Java 的==和equals比较

Java8 新特性和改进总览

Java assertion介绍和用法

Java 类的生命周期详解

Java 数组转型和范型

Java 之String类型

Java 值传递与引用传递

Java 内存模型及GC原理

Spring 常见技术问题总结

Spring 编程技术的经典面试题

Apache httpd.conf 配置详解