Java Generics - Рекомендации по использованию подстановочных знаков

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

  • Upper bound Wildcard-? расширяет Тип.

  • Lower bound Wildcard-? супер Тип.

  • Unbounded Wildcard -?

Чтобы решить, какой тип подстановочного знака лучше всего подходит для данного условия, давайте сначала классифицируем тип параметров, передаваемых в метод, как in и out параметр.

  • in variable- Переменная in предоставляет данные коду. Например, copy (src, dest). Здесь src действует как переменная, являющаяся копируемыми данными.

  • out variable- Переменная out содержит данные, обновленные кодом. Например, copy (src, dest). Здесь dest действует как переменная, скопировав данные.

Рекомендации по использованию подстановочных знаков.

  • Upper bound wildcard - Если переменная имеет in категории используйте ключевое слово extends с подстановочным знаком.

  • Lower bound wildcard - Если переменная имеет out категория, используйте ключевое слово super с подстановочным знаком.

  • Unbounded wildcard - Если к переменной можно получить доступ с помощью метода класса объекта, используйте несвязанный подстановочный знак.

  • No wildcard - Если код обращается к переменной в обоих in и out категория, то не используйте подстановочные знаки.

пример

Следующий пример иллюстрирует вышеупомянутые концепции.

package com.tutorialspoint;

import java.util.ArrayList;
import java.util.List;

public class GenericsTester {

   //Upper bound wildcard
   //in category
   public static void deleteCat(List<? extends Cat> catList, Cat cat) {
      catList.remove(cat);
      System.out.println("Cat Removed");
   }

   //Lower bound wildcard
   //out category
   public static void addCat(List<? super RedCat> catList) {
      catList.add(new RedCat("Red Cat"));
      System.out.println("Cat Added");
   }

   //Unbounded wildcard
   //Using Object method toString()
   public static void printAll(List<?> list) {
      for (Object item : list)
         System.out.println(item + " ");
   }

   public static void main(String[] args) {

      List<Animal> animalList= new ArrayList<Animal>();
      List<RedCat> redCatList= new ArrayList<RedCat>();

      //add list of super class Animal of Cat class
      addCat(animalList);
      //add list of Cat class
      addCat(redCatList);  
      addCat(redCatList);  

      //print all animals
      printAll(animalList);
      printAll(redCatList);

      Cat cat = redCatList.get(0);
      //delete cat
      deleteCat(redCatList, cat);
      printAll(redCatList); 
   }
}

class Animal {
   String name;
   Animal(String name) { 
      this.name = name;
   }
   public String toString() { 
      return name;
   }
}

class Cat extends Animal { 
   Cat(String name) {
      super(name);
   }
}

class RedCat extends Cat {
   RedCat(String name) {
      super(name);
   }
}

class Dog extends Animal {
   Dog(String name) {
      super(name);
   }
}

Это даст следующий результат -

Cat Added
Cat Added
Cat Added
Red Cat 
Red Cat 
Red Cat 
Cat Removed
Red Cat