Java generic field declaration

2019-02-21 17:45发布

问题:

In a class without generic types I want to declare a rather complex generic field similar to these:

public class Client {
    private Map<Class<T extends Serializable>, List<Consumer<S extends T>>> classToConsumerTry1;

    private <T extends Serializable, S extends T> Map<Class<T>, List<Consumer<S>>> classToConsumerTry2;
}

promblem is the java compiler won't let me :)

So my question is how do I correctly introduce T and S without adding types to my class Client.

My goal is to enforce the Class being a subtype of Serializable and the Consumer being a subtype of the class you chose for Class.

回答1:

You can't. Your only option is to declare the generic type parameters in your Client class declaration. If your Client class has no generic type parameters, its members can't be generic. You must use actual types in the declaration of your class members.



回答2:

You have to somewhere introduce the type-parameter, so that you can use them in the definition for your class members.

Introducing a type-parameter can be done only on a class-level, or on a method-level. In your case, it should be on class-level:

public class Client<T extends Serializable, S extends T> {
    private Map<Class<T>, List<Consumer<S>>> classToConsumerTry1;

    private Map<Class<T>, List<Consumer<S>>> classToConsumerTry2;
}

This, however, implies that for both members (classToConsumerTry1 and classToConsumerTry2), T and S are the same. If you want them to be different, the you will have to get these two values from two different classes, both of which are parameterized with separate type-parameters.



回答3:

You can not do it directly on the field, but you can maintain your types policy by generic methods (they can exist even in non-generic class):

@SuppressWarnings("rawtypes")
private Map<Class, List> map;

public <T extends Serializable> void put(Class<T> key, List<Consumer<? extends T>> value) {
    map.put(key, value);
}

@SuppressWarnings("unchecked")
public <T extends Serializable> List<Consumer<? extends T>> get(Class<T> key) {
    return map.get(key);
}


回答4:

You can't the way you want it done. But you can set as an Object and cast. Other hacky answers include using a List of <? extends Serializable> (but this only works for one extends) and then add() your item or an internal class that does something like:

private class Foo<T> {
        T obj;
}

so you can access/mutate obj internally.