Integer是int对应的包装类,它有一个int类型的字段存储数据,并且提供了基本操作,比如数学运算、int和字符串之间转换等。在Java5中,引入了自动装箱和自动拆箱功能,Java可以根据上下文,自动进行转换,极大地简化了相关编程。在Java5中新增了静态工厂方法valueOf,在调用它的时候会利用一个缓存机制,带来了明显的性能改进,这个值的默认缓存是-128到127之间。
像前面提到的整数,javac替我们自动把装箱转换为Integer.valueOf(),把拆箱替换为Integer.intValue(),这似乎也顺道回答了另一个问题,既然调用的是Integer.valueOf(),自然能够得到缓存的好处。
这种缓存机制并不是只有Integer才有,同样存在于其他的一些包装类,比如:
* Boolean,缓存了true/false对应的实例
* Short,同样是缓存了-128到127之间的数值
* Byte,数值有限,所以全部都被缓存
* Character,缓存范围’\u0000’到‘\u007F’
Integer的默认缓存是-128到127之间,但是在特别的应用场景,比如我们明确知道应用会频繁使用更大的数值,这时候该怎么办呢?
缓存上限值实际是可以根据需要调整的,JVM提供了参数设置:
-XX:AutoBoxCacheMax=N
int与integer的基本使用对比:
(1)Integer是int的包装类;int是基本数据类型;
(2)Integer变量必须实例化后才能使用;int变量不需要;
(3)Integer实际是对象的引用,指向此new的Integer对象;int是直接存储数据值;
(4)Integer的默认值是null;int的默认值是0。
int与Integer的深入对比:
(1)由于Integer变量实际上是对一个Integer对象的引用,所以两个通过new生成的Integer变量永远是不相等的(因为new生成的是两个对象,其内存地址不同)。
(2)Integer变量和int变量比较时,只要两个变量的值是相等的,则结果为true(因为包装类Integer和基本数据类型int比较时,java会自动拆包装为int,然后进行比较,实际上就变为两个int变量的比较)
(3)非new生成的Integer变量和new Integer()生成的变量比较时,结果为false。(因为非new生成的Integer变量指向的是java常量池中的对象【在-128至127范围内】,而new Integer()生成的变量指向堆中新建的对象,两者在内存中的地址不同)
(4)对于两个非new生成的Integer对象,进行比较时,如果两个变量的值在区间-128到127之间,则比较结果为true,如果两个变量的值不在此区间,则比较结果为false
valueOf() 方法的实现比较简单,就是先判断值是否在缓存池中,如果在的话就直接返回缓存池的内容。
public static Integer valueOf(int i) {
if (i >=IntegerCache.low && i <= IntegerCache.high)
returnIntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
在 Java 8 中,Integer 缓存池的大小默认为 -128~127。
原始数据类型线程安全:
原始数据类型的变量,显然要使用并发相关手段,才能保证线程安全,如果有线程安全的计算需要,建议考虑使用类似AtomicInteger、AtomicLong这样的线程安全类。
原始数据类型和引用类型局限性:
原始数据类型和Java泛型并不能配合使用;也是基于此设计的自动装箱/拆箱操作
无法高效的表达数据,也不便于表达复杂的数据结构,比如vector和tuple。