今天在后端接口调试过程中,发现一个奇怪问题:
if(team.userId==userId){...}
断点中的连个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, 简单记录下, 欢迎踩坑!