Java Generics - Wskazówki dotyczące używania symboli wieloznacznych
Symbole wieloznaczne mogą być używane na trzy sposoby -
Upper bound Wildcard-? rozszerza Type.
Lower bound Wildcard-? super Type.
Unbounded Wildcard -?
Aby zdecydować, który typ symbolu wieloznacznego najlepiej pasuje do warunku, najpierw sklasyfikujmy typ parametrów przekazywanych do metody jako in i out parametr.
in variable- Zmienna in dostarcza dane do kodu. Na przykład copy (src, dest). Tutaj src działa jak w zmiennej będącej danymi do skopiowania.
out variable- Zmienna out przechowuje dane aktualizowane przez kod. Na przykład copy (src, dest). Tutaj dest działa jak w zmiennej po skopiowaniu danych.
Wytyczne dotyczące symboli wieloznacznych.
Upper bound wildcard - Jeśli zmienna ma wartość in category, użyj słowa kluczowego extends z symbolem wieloznacznym.
Lower bound wildcard - Jeśli zmienna ma wartość out kategorii, użyj słowa kluczowego super z symbolem wieloznacznym.
Unbounded wildcard - Jeśli dostęp do zmiennej można uzyskać za pomocą metody klasy Object, użyj niezwiązanego symbolu wieloznacznego.
No wildcard - Jeśli kod uzyskuje dostęp do zmiennej w obu in i out category, nie używaj symboli wieloznacznych.
Przykład
Poniższy przykład ilustruje powyższe koncepcje.
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);
}
}
To da następujący wynik -
Cat Added
Cat Added
Cat Added
Red Cat
Red Cat
Red Cat
Cat Removed
Red Cat