Static Method in Interface with Generic signature

后端 未结 3 1659
你的背包
你的背包 2021-02-08 14:55

As of Java 8 you can have default or static methods implemented in Interfaces as the below

public interface DbValuesEnumIface>          


        
相关标签:
3条回答
  • 2021-02-08 15:16

    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>
    
    0 讨论(0)
  • 2021-02-08 15:19

    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);

    0 讨论(0)
  • 2021-02-08 15:32

    You can change from static to default and it will compile successfully.

    default EvaluationStatuses fromId(Integer id)

    0 讨论(0)
提交回复
热议问题