[Java开发之路](19)Long缓存问题

Posted SunnyYoona

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了[Java开发之路](19)Long缓存问题相关的知识,希望对你有一定的参考价值。

Long中有个小小的陷阱,就是在-128至127范围内,Long.valueOf(long l)返回的Long的实例是相同的,而在此范围之外每次使用valueOf(long l)时,返回的实例都是不同的。

举例:

  1. System.out.println(Long.valueOf(-129) == Long.valueOf(-129)); // false
  2. System.out.println(Long.valueOf(-128) == Long.valueOf(-128)); // true
  3. System.out.println(Long.valueOf(127) == Long.valueOf(127)); // true
  4. System.out.println(Long.valueOf(128) == Long.valueOf(128)); // false
下面我们通过Long源码进行分析一下:
  1. public static Long valueOf(long l) {
  2. final int offset = 128;
  3. if (l >= -128 && l <= 127) { // will cache
  4. return LongCache.cache[(int)l + offset];
  5. }
  6. return new Long(l);
  7. }

从上面代码中我们可以看出先判断传递过来的数值是否在[-128,127]之间,如果是则直接从缓存中返回对应的引用,否则新创建一个Long的实例。所以说如果不在这个区间范围内,返回一个新创建的Long类型引用,用==判断就会理所当然的返回false,地址不一样。但是如果我们使用equals方法,则会返回true,数值是一样的。

  1. Long.valueOf(128).equals(Long.valueOf(128)) // true

我们看看对于在区间范围之内,是如何返回对应的引用?最重要的是Long类中有一个静态的内部类LongCache,专门用于缓存-128至127之间的值。

  1. private static class LongCache {
  2. private LongCache(){}
  3. static final Long cache[] = new Long[-(-128) + 127 + 1];
  4. static {
  5. for(int i = 0; i < cache.length; i++)
  6. cache[i] = new Long(i - 128);
  7. }
  8. }

在LongCache类中定义了一个cache数组,来存储缓存数据。我们可以看到cache数组的长度:-(-128) + 127 + 1,很明了的知道缓存数据从-128到127,后面的1代表数字0,一共256个元素。

valueOf这个方法设计比较好的一点是offset,它的初始值设为128,目的就是为了数组下标128处存放0,这样就将正数和负数分隔开。



以上是关于[Java开发之路](19)Long缓存问题的主要内容,如果未能解决你的问题,请参考以下文章

开发成长之路(19)-- 缓存中间件:redis

[原创]java WEB学习笔记59:Struts2学习之路---OGNL,值栈,读取对象栈中的对象的属性,读取 Context Map 里的对象的属性,调用字段和方法,数组,list,map(代码片

web专家之路,知识沉淀

Java 从入门到进阶之路(十九)

Java中判断两个Long类型是否相等

[原创]java WEB学习笔记93:Hibernate学习之路---Hibernate 缓存介绍,缓存级别,使用二级缓存的情况,二级缓存的架构集合缓存,二级缓存的并发策略,实现步骤,集合缓存,查询缓