Принципы PECS в Java Generics

Java задняя часть переводчик Apple

Основы дженериков

泛型类
Сначала мы определяем простой класс Container:

public class Container {
    private String object;
    public void set(String object) { this.object = object; }
    public String get() { return object; }
}

Это наиболее распространенная практика. Одним из недостатков этого является то, что в контейнер можно загружать только элементы типа String. В будущем, если нам потребуется загрузить другие типы элементов, такие как Integer, мы должны переписать другой контейнер. Чтобы повторно использовать , использование дженериков может очень хорошо решить эту проблему.

public class Container<T> {
    // T stands for "Type"
    private T t;
    public void set(T t) { this.t = t; }
    public T get() { return t; }
}

Чтобы наш класс Container можно было использовать повторно, мы можем заменить T любым типом, который захотим:

Container<Integer> integerContainer = new Container<Integer>();
Container<Double> doubleContainer = new Container<Double>();
Container<String> stringContainer = newContainer<String>();

泛型方法
После прочтения универсальных классов давайте взглянем на универсальные методы. Объявление универсального метода так же просто, как добавление префикса возвращаемого типа в виде :

public class Util {
    public static <K, V> boolean compare(Pair<K, V> p1, Pair<K, V> p2) {
        return p1.getKey().equals(p2.getKey()) &&
           p1.getValue().equals(p2.getValue());
    }
}
public class Pair<K, V> {
    private K key;
    private V value;
    public Pair(K key, V value) {
        this.key = key;
        this.value = value;
    }
    public void setKey(K key) { this.key = key; }
    public void setValue(V value) { this.value = value; }
    public K getKey()   { return key; }
    public V getValue() { return value; }
}

Мы можем вызывать универсальные методы следующим образом:

Pair<Integer, String> p1 = new Pair<>(1, "apple");
Pair<Integer, String> p2 = new Pair<>(2, "pear");
boolean same = Util.<Integer, String>compare(p1, p2);

Или после Java 1.7 вы можете использовать вывод типа, чтобы позволить Java автоматически выводить соответствующие параметры типа:

Pair<Integer, String> p1 = new Pair<>(1, "apple");
Pair<Integer, String> p2 = new Pair<>(2, "pear");
boolean same = Util.compare(p1, p2);

泛型接口
Определение и использование универсального интерфейса и универсального класса в основном одинаковы. Общие интерфейсы часто используются в производителях различных классов, см. пример:

//定义一个泛型接口
public interface Generator<T> {
    public T next();
}

Когда класс, реализующий универсальный интерфейс, не передает общие аргументы:

/**
 * 未传入泛型实参时,与泛型类的定义相同,在声明类的时候,需将泛型的声明也一起加到类中
 * 即:class FruitGenerator<T> implements Generator<T>{
 * 如果不声明泛型,如:class FruitGenerator implements Generator<T>,编译器会报错:"Unknown class"
 */
 class FruitGenerator<T> implements Generator<T>{
       @Override
       public T next() {
            return null;
        }
}

Когда класс, реализующий универсальный интерфейс, передает общие аргументы:

/**
 * 传入泛型实参时:
 * 定义一个生产器实现这个接口,虽然我们只创建了一个泛型接口Generator<T>
 * 但是我们可以为T传入无数个实参,形成无数种类型的Generator接口。
 * 在实现类实现泛型接口时,如已将泛型类型传入实参类型,则所有使用泛型的地方都要替换成传入的实参类型
 * 即:Generator<T>,public T next();中的的T都要替换成传入的String类型。
 */
public class FruitGenerator implements Generator<String> {

    private String[] fruits = new String[]{"Apple", "Banana", "Pear"};

    @Override
    public String next() {
        Random rand = new Random();
        return fruits[rand.nextInt(3)];
    }
}

граничный символ

Теперь мы хотим реализовать такую ​​функцию, чтобы найти количество элементов, превышающее определенный элемент в универсальном массиве, мы можем сделать это следующим образом:

public static <T> int countGreaterThan(T[] anArray, T elem) {
    int count = 0;
    for (T e : anArray)
        if (e > elem)  // compiler error
            ++count;
    return count;
}

Но это заведомо неправильно, потому что кроме примитивных типов типа short, int, double, long, float, byte, char другие классы могут не использовать оператор >, поэтому компилятор сообщает об ошибке, так как же решить эту проблему? ? Ответ заключается в использовании граничных символов.

public interface Comparable<T> {
    public int compareTo(T o);
}

Создание объявления, подобного приведенному ниже, эквивалентно указанию компилятору, что параметр типа T представляет все классы, реализующие интерфейс Comparable, что эквивалентно указанию компилятору на то, что все они реализуют по крайней мере метод compareTo.

public static <T extends Comparable<T>> int countGreaterThan(T[] anArray, T elem) {
    int count = 0;
    for (T e : anArray)
        if (e.compareTo(elem) > 0)
            ++count;
    return count;
}

подстановочный знак

Прежде чем понять подстановочные знаки, мы должны сначала прояснить концепцию или позаимствовать класс Container, который мы определили выше, предполагая, что мы добавим такой метод:

public void boxTest(Container<Number> n) { /* ... */ }

Итак, теперь Контейнер Какие типы параметров разрешено принимать? Можем ли мы пройти в контейнере или Контейнер Шерстяная ткань? Ответ — нет, хотя Integer и Double являются подклассами Number, но в общем Container или Контейнер с контейнером Между ними нет никаких отношений. Это очень важно, и давайте углубим наше понимание на полном примере.

Во-первых, давайте определим несколько простых классов, которые мы будем использовать ниже:

class Fruit {}
class Apple extends Fruit {}
class Orange extends Fruit {}

В следующем примере создается универсальный класс Reader, а затем в f1(), когда мы пытаемся Fruit f = fruitReader.readExact(apples); компилятор сообщит об ошибке, потому что List со списком Между ними нет никаких отношений.

public class GenericReading {
    static List<Apple> apples = Arrays.asList(new Apple());
    static List<Fruit> fruit = Arrays.asList(new Fruit());
    static class Reader<T> {
        T readExact(List<T> list) {
            return list.get(0);
        }
    }
    static void f1() {
        Reader<Fruit> fruitReader = new Reader<Fruit>();
        // Errors: List<Fruit> cannot be applied to List<Apple>.
        // Fruit f = fruitReader.readExact(apples);
    }
    public static void main(String[] args) {
        f1();
    }
}

Но согласно нашим привычным привычкам мышления должна быть связь между Apple и Fruit, но компилятор не может ее распознать, так как же решить эту проблему в универсальном коде? Мы можем исправить это, используя подстановочные знаки:

static class CovariantReader<T> {
    T readCovariant(List<? extends T> list) {
        return list.get(0);
    }
}
static void f2() {
    CovariantReader<Fruit> fruitReader = new CovariantReader<Fruit>();
    Fruit f = fruitReader.readCovariant(fruit);
    Fruit a = fruitReader.readCovariant(apples);
}
public static void main(String[] args) {
    f2();
}

Это эквивалентно указанию компилятору, что параметры, принимаемые методом readCovariant класса fruitReader, имеют длину, равную длине подклассов, удовлетворяющих Fruit (включая сам Fruit), так что отношения между подклассом и родительским классом также связаны.

принципы PECS

Выше мы видели использование, похожее на , с помощью которого мы можем получить элементы из списка, поэтому можем ли мы добавлять элементы в список? Давайте попробуем:

public class GenericsAndCovariance {
    public static void main(String[] args) {
        // Wildcards allow covariance:
        List<? extends Fruit> flist = new ArrayList<Apple>();
        // Compile Error: can't add any type of object:
        // flist.add(new Apple())
        // flist.add(new Orange())
        // flist.add(new Fruit())
        // flist.add(new Object())
        flist.add(null); // Legal but uninteresting
        // We Know that it returns at least Fruit:
        Fruit f = flist.get(0);
    }
}

Ответ - нет, компилятор не позволяет нам это сделать, почему? Для этой проблемы мы могли бы также рассмотреть ее с точки зрения компилятора. Поскольку список List сам по себе может иметь несколько значений:

List<? extends Fruit> flist = new ArrayList<Fruit>();
List<? extends Fruit> flist = new ArrayList<Apple>();
List<? extends Fruit> flist = new ArrayList<Orange>();

Когда мы пытаемся добавить Apple, flist может указывать на новый ArrayList. ();
Когда мы пытаемся добавить Orange, flist может указывать на новый ArrayList. ();
Когда мы пытаемся добавить Fruit, этот Fruit может быть любым типом Fruit, а flist может хотеть только определенный тип Fruit, компилятор не может его распознать, поэтому он сообщит об ошибке.
Следовательно, для класса коллекции, который реализует , он может рассматриваться только как источник, который предоставляет (получает) элементы извне, и не может использоваться как потребитель для получения (добавления) элементов извне.

Что, если мы хотим добавить элементы? Вы можете использовать :

public class GenericWriting {
    static List<Apple> apples = new ArrayList<Apple>();
    static List<Fruit> fruit = new ArrayList<Fruit>();
    static <T> void writeExact(List<T> list, T item) {
        list.add(item);
    }
    static void f1() {
        writeExact(apples, new Apple());
        writeExact(fruit, new Apple());
    }
    static <T> void writeWithWildcard(List<? super T> list, T item) {
        list.add(item)
    }
    static void f2() {
        writeWithWildcard(apples, new Apple());
        writeWithWildcard(fruit, new Apple());
    }
    public static void main(String[] args) {
        f1(); 
        f2();
    }
}

Таким образом мы можем добавлять элементы в контейнер, но недостаток использования super в том, что мы не можем в дальнейшем получить элементы в контейнере.Причина очень проста.Продолжим рассматривать эту проблему с позиции компилятора , Для списка List это может иметь следующие значения:

List<? super Apple> list = new ArrayList<Apple>();
List<? super Apple> list = new ArrayList<Fruit>();
List<? super Apple> list = new ArrayList<Object>();

Когда мы пытаемся получить яблоко через список, мы можем получить фрукт, который может быть апельсином и другими типами фруктов.

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

«Producer Extends» — если вам нужен доступный только для чтения список, из которого можно создать T, используйте .
«Consumer Super» — если вам нужен список только для записи для потребления T, используйте .
Если нам нужно читать и писать одновременно, мы не можем использовать подстановочные знаки.

hack

Если вы прочитаете исходный код некоторых классов коллекций Java, вы обнаружите, что они иногда используются вместе, например:

public class Collections {
    public static <T> void copy(List<? super T> dest, List<? extends T> src) {
        for (int i=0; i<src.size(); i++)
            dest.set(i, src.get(i));
    }
}

Так что иногда вы все еще хотите нарушить рутину и хотите писать и читать одновременно, вы можете использовать описанный выше метод:
例子

public class PECSTest {

    private List<? extends Father> glist = new ArrayList<>();

    public static void main(String[] args) {
        PECSTest pecsTest = new PECSTest();
        pecsTest.test();
    }

    private void test() {
        List<Father> list = new ArrayList<>();
        Collections.copy(list, this.glist);
        list.add(new Father("father"));  // 添加父类
        list.add(new Child(23));  // 添加一个子类
        glist = new ArrayList<>(list);
        System.out.println(glist);     // 打印
    }

    class Father {

        public String name;

        public Father(String name) {
            this.name = name;
        }

        @Override
        public String toString() {
            return "Father{" +
                    "name='" + name + '\'' +
                    '}';
        }
    }

    class Child extends Father {
        public int age;

        public Child(int age) {
            super("child");
            this.age = age;
        }

        @Override
        public String toString() {
            return "Child{" +
                    "name='" + name + '\'' +
                    ", age=" + age +
                    '}';
        }
    }
}

输出

[Father{name='father'}, Child{name='child', age=23}]