先看一段代码:

package com.test;

import java.util.Scanner;

public class IntegerCache {
public static void main(String[] args) {
Scanner input = new Scanner(System.in);
while(input.hasNextInt()){
int ii = input.nextInt();
System.out.println("===" + ii + "的相等判断 ===");
//通过两个new产生的Integer对象
Integer i = new Integer(ii);
Integer j = new Integer(ii);
System.out.println("new的对象:" + (i == j)); //基本类型转换为包装类型后比较
i = ii;
j = ii;
System.out.println("基本类型转换的对象:" + (i == j)); //通过静态方法生成的一个实例
i= Integer.valueOf(ii);
j = Integer.valueOf(ii);
System.out.println("valueOf的对象:" + (i == j));
}
}
}

结果:

看看不同数据的测试结果,如果你感兴趣可以测试一下其他的数据,最后发现-128 到 127 基础类型转化的对象和valueOf转化的对象 == 是 true

下面解释一下原因:

1、new产生的Integer对象

  new声明的就是要生成一个新的对象,2个对象比较内存地址肯定不相等,比较结果为false

2、装箱生成的对象

  对于这一点首先要说明的是装箱动作是通过Integer.valueOf方法进行的。

  Integer i = 100; (注意:不是 int i = 100; )

  实际上,执行上面那句代码的时候,系统为我们执行了:Integer i = Integer.valueOf(100); 此即基本数据类型的自动装箱功能。

//valueOf如何生成对象:

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

这是JDK的源码,low=-128,h=127,这段代码意为如果是-128到127之间的int类型转换为Integer对象,则直接从IntegerCache里获取,来看看IntegerCache这个类

private static class IntegerCache {
static final int low = -128;
static final int high;
// 内部静态数组,容纳-128到127之间的对象
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() {}
}

结论:

在创建Integer对象,尽量少用new创建,尽量使用valueOf,利用整型池的提高系统性能,通过包装类的valueOf生成包装实例可以提高空间和时间性能。

另外:在判断对象是否相等的时候尽量使用equals方法,避免使用“==”产生非预期结果。

附1:

Integer 可以通过参数改变缓存范围(附:最大值 127 可以通过 JVM 的启动参数 -XX:AutoBoxCacheMax=size 修改)

-XX:AutoBoxCacheMax 这个参数是设置Integer缓存上限的参数。

理论上讲,当系统需要频繁使用Integer时,或者说堆内存中存在大量的Integer对象时,可以考虑提高Integer缓存上限,避免JVM重复创造对象,提高内存的使用率,减少GC的频率,从而提高系统的性能

理论归理论,这个参数能否提高系统系统关键还是要看堆中Integer对象到底有多少、以及Integer的创建的方式,如果堆中的Integer对象很少,重新设置这个参数并不会提高系统的性能。

即使堆中存在大量的Integer对象,也要看Integer对象是如何产生的

1. 大部分Integer对象通过Integer.valueOf()产生。说明代码里存在大量的拆箱与装箱操作。这时候设置这个参数会系统性能有所提高

2. 大部分Integer对象通过反射,new产生。这时候Integer对象的产生大部分不会走valueOf()方法,所以设置这个参数也是无济于事

附2:

其他缓存的对象:

这种缓存行为不仅适用于Integer对象。我们针对所有整数类型的类都有类似的缓存机制。

Byte 有 ByteCache 用于缓存 Byte 对象(-128 到 127)

Short 有 ShortCache 用于缓存 Short 对象(-128 到 127)

Long 有 LongCache 用于缓存 Long 对象(-128 到 127)

Character 有 CharacterCache 用于缓存 Character 对象(0 到 127)

Float 没有缓存

Doulbe 没有缓存

除了 Integer 可以通过参数改变范围外,其它的都不行。

附3:

关于垃圾回收器:

Integer i = 127;

i = null;   //不会被垃圾回收器回收,这里的代码不会有对象符合垃圾回收器的条件,i 虽然被赋予null,但它之前指向的是cache中的Integer对象,而cache没有被赋null,所以Integer 127这个对象还是存在的

Integer i = 200;

i = null;  //会被垃圾回收器回收,而如果 i 大于127或小于-128,则它所指向的对象将符合垃圾回收的条件

Java之戳中痛点 - (7)善用Java整型缓存池的更多相关文章

  1. Java之戳中痛点 - (4)i++ 和 ++i 探究原理

    先看一个例子: package com.test; public class AutoIncrement { public static void main(String[] args) { int ...

  2. Java之戳中痛点 - (1)易变业务使用脚本语言编写

    脚本语言的3大特征: 1.灵活:脚本语言一般是动态类型,可以不声明变量类型直接使用,也可以在运行期改变类型:2.便捷:脚本语言是解释性语言,在运行期变更非常方便,而不用重启服务3.简单:脚本语言语法比 ...

  3. Java之戳中痛点 - (6)避免类型自动转换,例如两个整数相除得浮点数遇坑

    先来看一个例子: package com.test; public class calculate { /** * 光速30万公里/秒 */ public static final int LIGHT ...

  4. Java之戳中痛点 - (8)synchronized深度解析

    概览: 简介:作用.地位.不控制并发的影响 用法:对象锁和类锁 多线程访问同步方法的7种情况 性质:可重入.不可中断 原理:加解锁原理.可重入原理.可见性原理 缺陷:效率低.不够灵活.无法预判是否成功 ...

  5. Java之戳中痛点 - (2)取余用偶判断,不要用奇判断

    取余判断原则:取余用偶判断,不要用奇判断 先看一个 程序: package com.test; import java.util.Scanner; public class t1 { public s ...

  6. Java之戳中痛点 - (5)switch语句break不能忘以及default不同位置的用法

    先看一段代码: public class Test{ public static void main(String[] args){ System.)); } } public static Stri ...

  7. Java之戳中痛点 - (3)三目运算符的两个操作数类型尽量一致

    先看一个例子: package com.test; public class TernaryOperator { public static void main(String[] args) { in ...

  8. java获取request中的参数、java解析URL问号后的参数

    java获取request中的参数.java解析URL问号后的参数.有时候我们需要从request中获取参数,或者获取拼接在Url后面的参数,有时候一个一个去拿有点麻烦,一起拿出来放在一个map里面需 ...

  9. 在RobotFramework--RIDE中把日期转化为整型进行运算

    在RobotFramework--RIDE中把日期转化为整型进行运算 运行结果: 20180906 16:10:17.919 : INFO : ${time} = 2018-09-06 16:10:1 ...

随机推荐

  1. 1. Skippr

    Skippr 是一个超级简单的 jQuery 幻灯片插件.只是包括你的网页中引入 jquery.skippr.css 和 jquery.skippr.js 文件就能使用了.Skippr 能够自适应窗口 ...

  2. bzoj3939 【USACO 2015 FEB GOLD 】cow hopscotch

    Description 就像人类喜欢玩"跳房子"的游戏,农民约翰的奶牛已经发明了该游戏的一个变种自己玩.由于笨拙的动物体重近一吨打,牛跳房子几乎总是以灾难告终,但这是没有阻止奶牛几 ...

  3. JVM 方法调用之动态分派

    1. 动态分派 一个体现是重写(override).下面的代码,运行结果很明显. public class App { public static void main(String[] args) { ...

  4. Linux Shell——函数的使用

    文/一介书生,一枚码农. scripts are for lazy people. 函数是存在内存里的一组代码的命名的元素.函数创建于脚本运行环境之中,并且可以执行. 函数的语法结构为: functi ...

  5. redis持久化快速回忆手册

    Redis提供的持久化机制: 1). RDB持久化:该机制是指在指定的时间间隔内将内存中的数据集快照写入磁盘. 2). AOF持久化:该机制将以日志的形式记录服务器所处理的每一个写操作,在Redis服 ...

  6. PHP:基于百度大脑api实现OCR文字识别

    有个项目要用到文字识别,网上找了很多资料,效果不是很好,偶然的机会,接触到百度大脑.百度大脑提供了很多解决方案,其中一个就是文字识别,百度提供了三种文字识别,分别是银行卡识别.身份证识别和通用文字识别 ...

  7. OC分类(Category)

    Category 分类 ,又称为类别.类目 概念 Category有多种翻译:分类.类别.类目(一般叫分类的多) Category式OC特有的语法,其他语言没有的语法(类似于C#语言中的"扩 ...

  8. C#集合之字典

    字典表示一种复杂的数据结构,这种数据结构允许按照某个键来访问元素.字典也称为映射或散列表. 字典的主要特性是能根据键快速查找值.也可以自由添加和删除元素,这有点像List<T>(http: ...

  9. javascript设计模式详解之策略模式

    接上篇命令模式来继续看下js设计模式中另一种常用的模式,策略模式.策略模式也是js开发中常用的一种实例,不要被这么略显深邃的名字给迷惑了.接下来我们慢慢看一下. 一.基本概念与使用场景: 基本概念:定 ...

  10. SG函数学(hua)习(shui)记录

    ---恢复内容开始--- 听说有一个东西叫SG函数 觉得自己好像原来是懂一些粗浅的应用但现在感觉要再深♂入一点呢 让我们先来介绍一下SG函数吧 这是某类满足下列条件的玄学博弈问题解法 双人.回合制: ...