java - 为什么我们不允许在java中重写枚举中的hashcode

标签 java enums hashcode

我现在只知道Java中没有Enum的hashCode()实现。它返回的只是 super.hashCode(),而其他不可变类(例如 String)都有自己的 hashCode() 实现。这使得 Enum 在不同 JVM 中使用时不安全。我认为ordinal()非常适合计算Enum的hashCode(),但是Enum中的hashCode()被定义为final,我无法覆盖它。我能想到的唯一解决方案是创建一个类似于 Enum 的全新类。有什么建议吗?


我想要实现这样的目标:

enum Fruit {
        APPLE, POMEGRANATE, KIWI;
        
        @override
        public int hashCode() {
            return Objects.hash(super.ordinal());
        }
        @override
        public boolean equals(Object obj) {
            if (this == obj)
                return true;
            if (obj == null)
                return false;
            if (getClass() != obj.getClass())
                return false;
            }
            Fruit other = (Fruit)obj;
            return super.ordinal() == other.ordinal();
        }
}

有什么解决办法吗?或者 我想了解为什么这个解决方案不好,因此在 java 中不允许。

最佳答案

你说:

This makes Enum unsafe when used across different JVM.

永远不能依赖任何对象上的hashCode来在JVM之间保持一致,如commented by shmosel .

Object##hashCode promise 的契约(Contract)明确指出您不能依赖 JVM 调用之间的 hashCode 相同。引用 Javadoc:

This integer need not remain consistent from one execution of an application to another execution of the same application.

你说:

I think ordinal() is perfect for calculating hashCode() of Enum

不,我想不是。您可以定义应用程序版本之间枚举元素的顺序。您甚至可以添加或删除元素。因此,我们通常不应该依赖应用程序版本之间的序号。

你说:

The only solution I can think of is to create a brand new class simalar to Enum.

我相信你会遇到麻烦。

你说:

Any suggestion?

是的。指定一个键。

如果您想标识 Enum 子类对象所表示的实体的值,我建议您向枚举类添加一个成员字段,以唯一地标识每个实例。

自然键

您分配的值取决于您的问题域。在特定行业或公司中,可能存在已知的标识符。例如,在处理水果时,也许可以指定一个拉丁科学植物学名称。

在关系数据库理论中,其名称是自然键

enum Fruit {
    APPLE ( "Malus domestica" ) , 
    POMEGRANATE ( "Punica granatum" ) , 
    KIWI ( "Actinidia deliciosa" );
    
    final String botanicalName ;
    
    Fruit ( String botanicalName ) 
    {
        this.botanicalName = botanicalName ;
    }
}

用法:

for( Fruit fruit : Fruit.values() )
{
    System.out.println( fruit + " = " + fruit.botanicalName ) ;
}

运行时。

APPLE = Malus domestica
POMEGRANATE = Punica granatum
KIWI = Actinidia deliciosa

查看此code run live at Ideone.com .

代理键(任意标识符)

如果您的业务域中不存在此类标识符,则分配任意永久标识符。

如果您不知道,请使用 universally unique identifier (UUID) 。请参阅UUID类。

在关系数据库理论中,我们称之为代理键。

enum Fruit {
    APPLE ( UUID.fromString( "9307e05e-b337-41e8-acec-a00645b00878" ) ) , 
    POMEGRANATE ( UUID.fromString( "6f67df08-b400-49af-94ed-e068ee58412f" ) ) , 
    KIWI ( UUID.fromString( "028c9e8a-9d25-48a1-b150-b892ea26807f" ) ) ;
    
    final UUID id ;
    
    Fruit ( UUID id ) 
    {
        this.id = id ;
    }
}

用法:

for( Fruit fruit : Fruit.values() )
{
    System.out.println( fruit + " ➔ " + fruit.id ) ;
}

运行时。

APPLE ➔ 9307e05e-b337-41e8-acec-a00645b00878
POMEGRANATE ➔ 6f67df08-b400-49af-94ed-e068ee58412f
KIWI ➔ 028c9e8a-9d25-48a1-b150-b892ea26807f

顺便说一句,“没有实现 EnumhashCode()”在技术上是不正确的。继承的实现就是它的实现。这个观点是 OOP 的基础。 .

关于java - 为什么我们不允许在java中重写枚举中的hashcode,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/72779912/

相关文章:

java - 在 Java 中缩短数组

java - ThriftSecurityException(用户 :root, 代码:BAD_CREDENTIALS)

java - 设计基于 key 的锁(或锁图)

c++ - 析构函数中的奇怪枚举

c# 获取所有大于给定值的枚举值?

java - AbstractMap 重写 hashcode 存在问题

java - 所有属性都必须影响 JavaBean 的 hashcode() 和 equals() 方法吗?

java - 为什么 HashMap 会重新哈希键对象提供的哈希码?

java - 整个代码私有(private)静态

java - 带有枚举字段的 Hibernate Search/Lucene 范围查询不返回任何结果