Java Generics с подстановочными знаками - PullRequest
2 голосов
/ 10 июля 2009

Есть какой-то способ исправить эту ситуацию (я постарался максимально упростить сценарий):

public class Test {

    public static void main(String[] args) {

        /*
         * HERE I would like to indicate that the CollectionGeneric can be of
         * something that extends Animal (but the constructor doesn't allow
         * wildcards)
         */
        CollectionGeneric<? extends Animal> animalsCollectionGeneric = new CollectionGeneric<Animal>();
        List<? extends Animal> animals = getAnimals();
        /* Why I cannt do that? */
        animalsCollectionGeneric.setBeans(animals);
    }

    private static List<? extends Animal> getAnimals() {
        return new ArrayList<Dog>();
    }
}

class CollectionGeneric<T> {
    private List<T> beans;

    public List<T> getBeans() {
        return (beans != null) ? beans : new ArrayList<T>();
    }

    public void setBeans(List<T> beans) {
        this.beans = beans;
    }
}

interface Animal {}

class Dog implements Animal{}

этот сценарий дает мне следующую ошибку:

The method setBeans(List<capture#2-of ? extends Animal>) in the type    
CollectionGeneric<capture#2-of ? extends Animal> is not applicable for
the arguments (List<capture#3-of ? extends Animal>)*

Я не уверен, есть ли способ сделать это с помощью дженериков,

1 Ответ

7 голосов
/ 10 июля 2009

Это означает, что две коллекции не могут иметь одинаковые границы типов:

    CollectionGeneric<? extends Animal> animalsCollectionGeneric = 
             new CollectionGeneric<Animal>(); 
    List<? extends Animal> animals = getAnimals()

Первый может иметь во время выполнения CollectionGeneric<Tiger>, а второй List<Gnu>. Смешивая их, вы потеряете безопасность типов (не говоря уже о бойне).

Поэтому вам нужно доказать компилятору, что эти два связаны, поэтому ваши общие подписи должны быть:

public void setBeans(List<? extends T> beans) {}
public List<T> getBeans();

и используется как:

List<? extends Animal> beans = getBeans();
GenericCollection<Animal> animals = new GenericCollection<Animal>();
animals.add(beans);
...