gpt4 book ai didi

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

转载 作者:行者123 更新时间:2023-12-02 18:08:43 26 4
gpt4 key购买 nike

我现在只知道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/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com