As of Java 8 you can have default or static methods implemented in Interfaces as the below
public interface DbValuesEnumIface>
There is no easy way as far as I can tell, first you need to change your method to default
, you can read more here of why you can't use generics in a static context.
But even if you change it to default
things are still not going to work, since you need to pass an instance or class type of the enum to that method, something like this:
public default T fromId(ID id, Class<T> t) {
if (id == null) {
return null;
}
for (T en : t.getEnumConstants()) {
// dome something
}
return null;
}
Now you are hitting another problem, inside fromId
- the only thing that you know is that T
extends an enum
- not your enum may be, thus getId
(which seems that your enums have) are simply not known by the compiler.
I don't know an easy way to make this work besides declaring an interface, like :
interface IID {
public int getId();
}
making your enum
implement it:
static enum My implements IID {
A {
@Override
public int getId() {
// TODO Auto-generated method stub
return 0;
}
};
}
and change the declaration to:
public interface DbValuesEnumIface<ID, T extends Enum<My> & IID>
Since there is no relationship between static
methods and the class’ type parameters, which describe how instances are parameterized, you have to make the static
method generic on its own. The tricky part is to get the declarations right to describe all needed constraints. And as this answer already explained, you need to a a Class
parameter, as otherwise, the implementation has no chance to get hands on the actual type arguments:
public interface DbValuesEnumIface<ID, T extends Enum<T>> {
public static
<ID, T extends Enum<T>&DbValuesEnumIface<ID,T>> T fromId(ID id, Class<T> type) {
if (id == null) {
return null;
}
for (T en : type.getEnumConstants()) {
if (en.getId().equals(id)) {
return en;
}
}
throw new NoSuchElementException();
}
ID getId();
String getDescriptionKey();
}
Note that the type parameters of the static
method are independent from the class’ type parameter. You may consider giving them different names for clarity.
So now, given you enum Statuses implements DbValuesEnumIface<Integer,Statuses>
example, you can use the method like Statuses status = DbValuesEnumIface.fromId(42, Statuses.class);
Note that for default
methods, it is possible to access the actual type, as a method providing the enum type will be provided by the implementation. You only have to declare the presence of the method within the interface
:
public interface DbValuesEnumIface<ID, T extends Enum<T>&DbValuesEnumIface<ID,T>> {
public default T fromId(ID id) {
if (id == null) {
return null;
}
for (T en : getDeclaringClass().getEnumConstants()) {
if (en.getId().equals(id)) {
return en;
}
}
throw new NoSuchElementException();
}
Class<T> getDeclaringClass();//no needed to implement it, inherited by java.lang.Enum
ID getId();
String getDescriptionKey();
}
However, the obvious disadvantage is that you need a target instance to invoke the method, i.e. Statuses status = Statuses.SOME_CONSTANT.fromId(42);
You can change from static
to default
and it will compile successfully.
default EvaluationStatuses fromId(Integer id)