Long类型比较过程中的小坑

今天在后端接口调试过程中,发现一个奇怪问题:

if(team.userId==userId){...}

Long类型比较过程中的小坑_第1张图片

 断点中的连个userId值都为1, 却不相等, 印象中-128-~127应该是相等的, 查看源码

private static class LongCache {  
  private LongCache(){}
  static final Long cache[] = new Long[-(-128) + 127 + 1];
  static {
    for(int i = 0; i < cache.length; i++)
      cache[i] = new Long(i - 128);
  }
}
public static Long valueOf(long l) {  
  final int offset = 128;
  if (l >= -128 && l <= 127) { // will cache
    return LongCache.cache[(int)l + offset];
  }
  return new Long(l);
}

Long中有一个静态的内部类LongCache,专门用于缓存-128至127之间的值,一共256个元素。它会判断传入的参数是否在-128-127之间, 如果是则直接从缓存中返回对应的引用,否则新创建一个Long的实例。  

即传值在-128~127之间, 返回的是同一个Long对象的引用, 故使用==时, 地址相等, 问题是断点中值确实再范围内

再看断点, 两个userId值虽然都为1, 但是内存地址却不一样

Long userId1 = new Long(1);  Long@11779   栈=>堆=>常量池

Long userId2 = new Long(1);  Long@11728   栈=>堆=>常量池

虽然都是指向同一个常量池对象, 但是堆引用地址不一样,内存地址值永远不会相等

Long userId1 = new Long(1);  Long@11779 

Long userId2 = 1L;  Long@12000

一个是堆引用,一个是直接指向常量池, 内存地址值永远不会相等

 所以, LongCache比较的应该是Long类型的变量才生效,  如

Long userId1 = 1L, Long userId2 = 1L  userId1 == userId2  --true

 实际开发中, 有各种业务代码比较, 类型并不固定, 推荐使用

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

Long重写了equeal方法, 比较值的时候, 拆箱使用基本数据类型比较

好了, 一个小bug, 简单记录下, 欢迎踩坑!

你可能感兴趣的:(java,数据结构,c#)