0

我有一个用例:

inteface A{
  get(EnumSet<?> fetchModes);
}
class B implements A{

  //Here FetchMode is an enum
  get(EnumSet<FetchMode> fetchMode){
   //Some logic here
  }
}

但它抛出编译时错误:

B 类的方法 get 与 A 类型的 get(EnumSet fetchMode) 具有相同的擦除,但不会覆盖它。

我读过 Enums 不能是通用的,但有没有办法实现这个用例?(基本上希望EnumSet是泛型的,不同的实现可以传递不同的Enums)

4

2 回答 2

5

如果参数类型完全匹配,则方法可以覆盖另一个方法,但您的方法不匹配。尽管EnumSet<FetchMode>是 的子类型EnumSet<?>,但它们并不完全相同。您没有覆盖get(EnumSet<?> fetchModes);子类中的超类方法,而是使用不同的参数类型重载它。当您检查或反编译字节码时,由于擦除,这两者具有相同的签名,get(EnumSet fetchModes)并且您的编译器开始抱怨。

这在 JLS §8.4.8.1中有介绍:

一个类不能有两个具有相同名称和类型擦除的成员方法

解决问题的一个天真的尝试是更改参数类型,使其与超类的参数类型兼容,在您的子类中正确覆盖该方法。

@Override
public void get(EnumSet<?> fetchModes) {

}

虽然这修复了编译器错误,但它仍然不够优雅,因为它允许您的 EnumSet 存储任何对象。但理想情况下,您可能希望它仅存储 Enum 的某些子类型。这个成语支持这一点。

您需要做的是声明一个具有有界类型参数的泛型接口,然后通过正确覆盖该方法来实现它,如下所示。

public interface A<E extends Enum<E>> {
    void get(EnumSet<E> fetchModes);
}

public class B implements A<FetchMode> {
    @Override
    public void get(EnumSet<FetchMode> fetchModes) {

    }
}
于 2020-03-10T07:34:25.747 回答
4

试试这个,你必须使泛型类型扩展 Enum:

public class B implements A<FetchMode> {
    //Here FetchMode is an enum
    public void get(EnumSet<FetchMode> fetchMode){
        //Some logic here
    }
}
}

interface A<T extends Enum<T>> {
    void get(EnumSet<T> fetchModes);
}
于 2020-03-10T07:29:07.283 回答