在Java中使用==而不是equals来比较不可变对象可以吗

2024-06-21

考虑调用静态工厂方法 valueOf 的两个 Integer 类型的引用,如下所示:-

    Integer a = Integer.valueOf("10"); 
    Integer b = Integer.valueOf("10"); 

考虑到Integer是不可变的,使用==而不是使用equals方法来比较a和b是否可以?我猜测 valueOf 方法确保只创建一个值为 10 的 Integer 实例,并且为每个使用值 10 创建的 Integer 返回对此实例的引用。

一般来说,是否可以使用 == 而不是 equals 来比较通过调用同一静态工厂方法创建的不可变类的两个引用?

编辑: Integer 类仅用作示例。我知道如果使用 == 进行比较,最大 127 的整数将返回 true。我需要知道的是,当我创建自己的不可变类时,使用 create() 方法说 MyImmutable ,它将确保不会创建重复的 MyImmutable 对象,如果我比较使用 create 方法创建的 2 个 MyImmutable 引用,可以吗?使用==而不是等于。


不,这通常不安全。这==运算符比较引用,而不是值。

Using ==恰好适用于 -128 到 127 之间的整数,但不适用于其他整数。下面的代码演示了==并不总是有效:

Integer a = Integer.valueOf(10); 
Integer b = Integer.valueOf(10); 
System.out.println(a == b);

true

Integer c = Integer.valueOf(1000); 
Integer d = Integer.valueOf(1000); 
System.out.println(c == d);

false

看看它在线工作:ideone http://ideone.com/rvcT1

这种行为的解释在于执行Integer.valueOf:

public static Integer valueOf(int i) {
     final int offset = 128;
     if (i >= -128 && i <= 127) { // must cache
         return IntegerCache.cache[i + offset];
     }
     return new Integer(i);
 }

source http://grepcode.com/file/repository.grepcode.com/java/root/jdk/openjdk/6-b14/java/lang/Integer.java#Integer.valueOf%28int%29

该标准还要求小输入(-128 到 127)的装箱整数为对象提供相同的引用。

5.1.7 拳击转换

如果装箱的值 p 为 true、false、一个字节、\u0000 到 \u007f 范围内的字符、或者 -128 到 127 之间的 int 或短数字,则令 r1 和 r2 为任意两个装箱转换的结果p。 r1 == r2 的情况总是如此。

然而,该标准对于超出此范围的整数没有做出这样的保证。


一般来说,是否可以使用 == 而不是 equals 来比较通过调用同一静态工厂方法创建的不可变类的两个引用?

如上图,一般情况下是行不通的。但是,如果您确保具有相同值的两个不可变对象始终具有相同的引用,那么是的,它可以工作。但是,您必须仔细遵守一些规则:

  • 构造函数不能是公共的。
  • 通过静态方法创建的每个对象都必须被缓存。
  • 每次要求您创建一个对象时,您必须首先检查缓存以查看是否已经创建了具有相同值的对象。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

在Java中使用==而不是equals来比较不可变对象可以吗 的相关文章

随机推荐