Zum Inhalt springen

„Generische Programmierung in Java“ – Versionsunterschied

aus Wikipedia, der freien Enzyklopädie
[gesichtete Version][gesichtete Version]
Inhalt gelöscht Inhalt hinzugefügt
Readability.
Zeile 18: Zeile 18:


Bisher war die ArrayList auf den Typ ''Object'' fixiert:
Bisher war die ArrayList auf den Typ ''Object'' fixiert:
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
List list = new ArrayList();
List list = new ArrayList();
list.add(new JButton("Button 1"));
list.add(new JButton("Button 1"));
list.add(new JButton("Button 2"));
list.add(new JButton("Button 2"));
list.add(new JButton("Button 3"));
list.add(new JButton("Button 3"));
list.add(new JButton("Button 4"));
list.add(new JButton("Button 4"));
list.add(new JButton("Button 5"));
list.add(new JButton("Button 5"));

for (int i = 0; i < list.size(); i++) {
for (int i = 0; i < list.size(); i++) {
JButton button = (JButton) list.get(i);
JButton button = (JButton) list.get(i);
button.setBackground(Color.white);
button.setBackground(Color.white);
}
}
</syntaxhighlight>
</syntaxhighlight>
Man beachte die notwendige [[Typumwandlung|explizite Typumwandlung]] (auch „Cast“ genannt) sowie die Typunsicherheit, die damit verbunden ist. Man könnte versehentlich ein Objekt in der <code>ArrayList</code> speichern, das keine Instanz der Klasse <code>JButton</code> ist. Die Information über den genauen Typ geht beim Einfügen in die Liste verloren, der Compiler kann also nicht verhindern, dass zur Laufzeit bei der expliziten Typumwandlung von <code>JButton</code> eine <code>ClassCastException</code> auftritt.
Man beachte die notwendige [[Typumwandlung|explizite Typumwandlung]] (auch „Cast“ genannt) sowie die Typunsicherheit, die damit verbunden ist. Man könnte versehentlich ein Objekt in der <code>ArrayList</code> speichern, das keine Instanz der Klasse <code>JButton</code> ist. Die Information über den genauen Typ geht beim Einfügen in die Liste verloren, der Compiler kann also nicht verhindern, dass zur Laufzeit bei der expliziten Typumwandlung von <code>JButton</code> eine <code>ClassCastException</code> auftritt.


Mit ''generischen Typen'' ist in Java Folgendes möglich:
Mit ''generischen Typen'' ist in Java Folgendes möglich:
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
List<JButton> list = new ArrayList<JButton>();
List<JButton> list = new ArrayList<JButton>();
list.add(new JButton("Button 1"));
list.add(new JButton("Button 1"));
list.add(new JButton("Button 2"));
list.add(new JButton("Button 2"));
list.add(new JButton("Button 3"));
list.add(new JButton("Button 3"));
list.add(new JButton("Button 4"));
list.add(new JButton("Button 4"));
list.add(new JButton("Button 5"));
list.add(new JButton("Button 5"));

for (int i = 0; i < list.size(); i++)
for (int i = 0; i < list.size(); i++)
list.get(i).setBackground(Color.white);
list.get(i).setBackground(Color.white);
</syntaxhighlight>
</syntaxhighlight>
Beim Auslesen ist nun keine explizite Typumwandlung mehr notwendig, beim Speichern ist es nur noch möglich, JButtons in der <code>ArrayList</code>&nbsp;''list'' abzulegen.
Beim Auslesen ist nun keine explizite Typumwandlung mehr notwendig, beim Speichern ist es nur noch möglich, JButtons in der <code>ArrayList</code>&nbsp;''list'' abzulegen.


Ab Java7 ist die Instanzierung generischer Typen vereinfacht worden. Die erste Zeile in obigem Beispiel kann seit Java 7 folgendermaßen geschrieben werden:
Ab Java7 ist die Instanzierung generischer Typen vereinfacht worden. Die erste Zeile in obigem Beispiel kann seit Java 7 folgendermaßen geschrieben werden:
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
List<JButton> list = new ArrayList<>();
List<JButton> list = new ArrayList<>();
</syntaxhighlight>
</syntaxhighlight>


Durch Kombination von generischen Typen mit den erweiterten For-Schleifen lässt sich obiges Beispiel kürzer fassen:
Durch Kombination von generischen Typen mit den erweiterten For-Schleifen lässt sich obiges Beispiel kürzer fassen:
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
List<JButton> list = new ArrayList<>();
List<JButton> list = new ArrayList<>();
list.add(new JButton("Button 1"));
list.add(new JButton("Button 1"));
list.add(new JButton("Button 2"));
list.add(new JButton("Button 2"));
list.add(new JButton("Button 3"));
list.add(new JButton("Button 3"));
list.add(new JButton("Button 4"));
list.add(new JButton("Button 4"));
list.add(new JButton("Button 5"));
list.add(new JButton("Button 5"));

for (JButton b: list)
for (JButton b: list)
b.setBackground(Color.white);
b.setBackground(Color.white);
</syntaxhighlight>
</syntaxhighlight>
Ein Beispiel für eine generische Klasse, die zwei Objekte von beliebigem, aber einander gleichem Typ beinhaltet, liefert der folgende Beispielcode:
Ein Beispiel für eine generische Klasse, die zwei Objekte von beliebigem, aber einander gleichem Typ beinhaltet, liefert der folgende Beispielcode:
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
public class DoubleObject<T> {
public class DoubleObject<T> {
private T object1;
private T object1;
private T object2;
private T object2;

public DoubleObject(T object1,T object2) {
public DoubleObject(T object1, T object2) {
this.object1 = object1;
this.object1 = object1;
this.object2 = object2;
this.object2 = object2;
}
}

public String toString() {
public String toString() {
return this.object1 + ", " + this.object2;
return this.object1 + ", " + this.object2;
}
}

public static void main(String[] args) {
public static void main(String[] args) {
DoubleObject<String> s = new DoubleObject<>("abc","def");
DoubleObject<String> s = new DoubleObject<>("abc", "def");
DoubleObject<Integer> i = new DoubleObject<>(123,456);
DoubleObject<Integer> i = new DoubleObject<>(123, 456);
System.out.println("DoubleObject<String> s=" + s.toString());
System.out.println("DoubleObject<String> s=" + s.toString());
System.out.println("DoubleObject<Integer> i=" + i.toString());
System.out.println("DoubleObject<Integer> i=" + i.toString());
}
}
}
}
</syntaxhighlight>
</syntaxhighlight>


Zeile 93: Zeile 93:
=== Invarianz ===
=== Invarianz ===
Bei Invarianz ist der Typparameter eindeutig. Damit bietet Invarianz die größtmögliche Freiheit bei der Benutzung des Typparameters. Beispielsweise sind für die Elemente einer ArrayList<Integer> alle Aktionen erlaubt, die auch bei der direkten Benutzung eines einzelnen Integers erlaubt sind (inklusive [[Autoboxing]]). Beispiel:
Bei Invarianz ist der Typparameter eindeutig. Damit bietet Invarianz die größtmögliche Freiheit bei der Benutzung des Typparameters. Beispielsweise sind für die Elemente einer ArrayList<Integer> alle Aktionen erlaubt, die auch bei der direkten Benutzung eines einzelnen Integers erlaubt sind (inklusive [[Autoboxing]]). Beispiel:
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
List<Integer> list = new ArrayList<Integer>();
List<Integer> list = new ArrayList<Integer>();
...
// ...
Integer x = list.get(index);
Integer x = list.get(index);
list.get(index).methodeVonInteger();
list.get(index).methodeVonInteger();
list.set(index, 98347); // Autoboxing, entspricht Integer.valueOf(98347)
list.set(index, 98347); // Autoboxing, entspricht Integer.valueOf(98347)
int y = list.get(index); // Auto-Unboxing
int y = list.get(index); // Auto-Unboxing
</syntaxhighlight>
</syntaxhighlight>
Diese Möglichkeiten werden mit wenig Flexibilität bei der [[Zuweisung]] von Objekten der generischen Klasse selbst erkauft. Beispielsweise ist Folgendes nicht erlaubt:
Diese Möglichkeiten werden mit wenig Flexibilität bei der [[Zuweisung]] von Objekten der generischen Klasse selbst erkauft. Beispielsweise ist Folgendes nicht erlaubt:
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
List<Number> list = new ArrayList<Integer>();
List<Number> list = new ArrayList<Integer>();
</syntaxhighlight>
</syntaxhighlight>
und das, obwohl Integer von Number abgeleitet ist. Der Grund liegt darin, dass der Compiler hier nicht mehr sicherstellen kann, dass keine Typfehler auftreten. Mit Arrays, die eine solche Zuweisung erlauben, hat man schlechte Erfahrungen gemacht:
und das, obwohl Integer von Number abgeleitet ist. Der Grund liegt darin, dass der Compiler hier nicht mehr sicherstellen kann, dass keine Typfehler auftreten. Mit Arrays, die eine solche Zuweisung erlauben, hat man schlechte Erfahrungen gemacht:
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
Number[] array = new Integer[10]; // OK, Integer[] ist abgeleitet von Number[]
// OK, Integer[] ist abgeleitet von Number[]
Number[] array = new Integer[10];
array[0] = new Double(5.0); // ArrayStoreException zur Laufzeit: Double -> Integer sind nicht zuweisungskompatibel

// ArrayStoreException zur Laufzeit: Double -> Integer sind nicht
// zuweisungskompatibel
array[0] = new Double(5.0);
</syntaxhighlight>
</syntaxhighlight>


Zeile 121: Zeile 125:


Referenzen müssen mit der Syntax '''? extends T''' explizit als kovariant gekennzeichnet werden. T heißt ''upper typebound'', also der allgemeinste Typparameter, der erlaubt ist.
Referenzen müssen mit der Syntax '''? extends T''' explizit als kovariant gekennzeichnet werden. T heißt ''upper typebound'', also der allgemeinste Typparameter, der erlaubt ist.
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
List<? extends Number> list;
List<? extends Number> list;
list = new ArrayList<Double>();
list = new ArrayList<Double>();
list = new ArrayList<Long>();
list = new ArrayList<Long>();
list = new ArrayList<Integer>();
list = new ArrayList<Integer>();

list.set(index, myInteger); // Typfehler vom Compiler
((List<Integer>)list).set(index, myInteger); // OK aber Warnung vom Compiler: unchecked cast
// Typfehler vom Compiler
list.set(index, myInteger);
// OK aber Warnung vom Compiler: unchecked cast
((List<Integer>) list).set(index, myInteger);
</syntaxhighlight>
</syntaxhighlight>
Das Ablegen von Elementen in diesen Listen ist nicht möglich, da dies, wie oben beschrieben, nicht typsicher ist (Ausnahme: '''null''' kann abgelegt werden). Bereits zur Kompilierzeit tritt ein Fehler auf. Allgemeiner gesagt, ist die Zuweisung
Das Ablegen von Elementen in diesen Listen ist nicht möglich, da dies, wie oben beschrieben, nicht typsicher ist (Ausnahme: '''null''' kann abgelegt werden). Bereits zur Kompilierzeit tritt ein Fehler auf. Allgemeiner gesagt, ist die Zuweisung
Zeile 134: Zeile 142:


Möglich dagegen ist das Auslesen von Elementen:
Möglich dagegen ist das Auslesen von Elementen:
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
Number n = list.get(index); // OK
Number n = list.get(index); // OK
Integer i = list.get(index); // Typfehler: Es muss sich bei '? extends Number' nicht um ein Integer handeln.
Integer i = list.get(index); // Typfehler: Es muss sich bei '? extends Number'
// nicht um ein Integer handeln.
Integer j = (Integer)list.get(index); // OK
Integer j = (Integer) list.get(index); // OK
</syntaxhighlight>
</syntaxhighlight>


Zeile 150: Zeile 159:
Kontravarianz bezeichnet das Verhalten der Vererbungshierarchie des generischen Typs entgegen der Hierarchie seines Typparameters. Übertragen auf das obige Beispiel würde das bedeuten: Eine Liste<Number> wäre zuweisungskompatibel zu einer Liste<Double>.
Kontravarianz bezeichnet das Verhalten der Vererbungshierarchie des generischen Typs entgegen der Hierarchie seines Typparameters. Übertragen auf das obige Beispiel würde das bedeuten: Eine Liste<Number> wäre zuweisungskompatibel zu einer Liste<Double>.
Dies wird folgendermaßen bewerkstelligt:
Dies wird folgendermaßen bewerkstelligt:
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
List<? super Double> list;
List<? super Double> list;
list = new ArrayList<Number>();
list = new ArrayList<Number>();
list = new ArrayList<Double>();
list = new ArrayList<Double>();
list = new ArrayList<Object>();
list = new ArrayList<Object>();
</syntaxhighlight>
</syntaxhighlight>


Ein Objekt, das sich kontravariant verhält, darf keine Annahmen darüber machen, inwiefern ein Element vom Typ V von T abgeleitet ist, wobei T der ''lower Typebound'' ist (im Beispiel von <code style="white-space: nowrap">? super Double</code> ist T <code>Double</code>).
Ein Objekt, das sich kontravariant verhält, darf keine Annahmen darüber machen, inwiefern ein Element vom Typ V von T abgeleitet ist, wobei T der ''lower Typebound'' ist (im Beispiel von <code style="white-space: nowrap">? super Double</code> ist T <code>Double</code>).
Deshalb kann aus den obigen Listen nicht gelesen werden:
Deshalb kann aus den obigen Listen nicht gelesen werden:
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
Number x = list.get(index); // Fehler: 'list' könnte vom Typ List<Object> sein
// Fehler: 'list' könnte vom Typ List<Object> sein
Double x = list.get(index); // Fehler: 'list' könnte List<Object> oder List<Number> sein
Number x = list.get(index);
// Fehler: 'list' könnte List<Object> oder List<Number> sein
Object x = list.get(index); // Die einzige Ausnahme: Objects sind auf jeden Fall in der Liste
Double x = list.get(index);
// Die einzige Ausnahme: Objects sind auf jeden Fall in der Liste
Object x = list.get(index);
</syntaxhighlight>
</syntaxhighlight>


Zeile 168: Zeile 182:


Unschwer zu erraten: Im Gegenzug kann in eine solche Liste ein Element abgelegt werden:
Unschwer zu erraten: Im Gegenzug kann in eine solche Liste ein Element abgelegt werden:
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
List<? super Number> list;
List<? super Number> list;
list.add(new Double(3.0)); // OK: 'list' hat immer den Typ List<Number>
list.add(new Double(3.0)); // OK: 'list' hat immer den Typ List<Number>
// oder List<Basisklasse von Number>. Damit
// oder List<Basisklasse von Number>. Damit
// ist die Zuweisung Double -> T immer erlaubt.
// ist die Zuweisung Double -> T immer erlaubt.
</syntaxhighlight>
</syntaxhighlight>


=== Uneingeschränkte parametrische Polymorphie ===
=== Uneingeschränkte parametrische Polymorphie ===
Zu guter Letzt bieten Generics noch gänzlich polymorphes Verhalten an. Hierbei kann keinerlei Aussage über die Typparameter gemacht werden, denn es wird in beide Richtungen keine Grenze angegeben. Dafür wurde die [[Wildcard (Java)|Wildcard]] definiert. Sie wird durch ein Fragezeichen repräsentiert.
Zu guter Letzt bieten Generics noch gänzlich polymorphes Verhalten an. Hierbei kann keinerlei Aussage über die Typparameter gemacht werden, denn es wird in beide Richtungen keine Grenze angegeben. Dafür wurde die [[Wildcard (Java)|Wildcard]] definiert. Sie wird durch ein Fragezeichen repräsentiert.
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
List<?> list;
List<?> list;
list = new ArrayList<Integer>();
list = new ArrayList<Integer>();
list = new ArrayList<Object>();
list = new ArrayList<Object>();
list = new ArrayList<String>();
list = new ArrayList<String>();
...
// ...
</syntaxhighlight>
</syntaxhighlight>
Der Typparameter selbst kann hierbei nicht genutzt werden, da keine Aussage möglich ist. Lediglich die Zuweisung T → Object ist erlaubt, da T auf jeden Fall ein Object ist. Im Gegenzug ist garantiert, dass der Code mit ''allen'' Ts arbeiten kann.
Der Typparameter selbst kann hierbei nicht genutzt werden, da keine Aussage möglich ist. Lediglich die Zuweisung T → Object ist erlaubt, da T auf jeden Fall ein Object ist. Im Gegenzug ist garantiert, dass der Code mit ''allen'' Ts arbeiten kann.


Nützlich kann so etwas sein, wenn man nur mit dem generischen Typ arbeitet:
Nützlich kann so etwas sein, wenn man nur mit dem generischen Typ arbeitet:
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
// Keine Informationen über den Typparameter nötig,
// Keine Informationen über den Typparameter nötig, kann ''beliebige'' Listen
// kann ''beliebige'' Listen aufnehmen.
// aufnehmen.
int readSize(List<?> list)
int readSize(List<?> list) {
return list.size();
{
}
return list.size();
}
</syntaxhighlight>
</syntaxhighlight>


Zur Verdeutlichung, dass hier Wildcards unnötig sind, und es eigentlich gar nicht um irgendeine Varianz geht, sei folgende Implementierung der obigen Funktion angegeben:
Zur Verdeutlichung, dass hier Wildcards unnötig sind, und es eigentlich gar nicht um irgendeine Varianz geht, sei folgende Implementierung der obigen Funktion angegeben:
<syntaxhighlight lang="java5">
<syntaxhighlight lang="java">
<T> int readSize(List<T> list)
<T> int readSize(List<T> list) {
return list.size();
{
}
return list.size();
}
</syntaxhighlight>
</syntaxhighlight>
<!--== Generische Arrays ==
<!--== Generische Arrays ==

Version vom 11. September 2019, 22:47 Uhr

Generische Programmierung in Java wird durch sog. Generics seit Java 1.5 ermöglicht. Der Begriff steht synonym für „parametrisierte Typen“. Die Idee dahinter ist, zusätzliche Variablen für Typen einzuführen. Diese Typ-Variablen repräsentieren zum Zeitpunkt der Implementierung unbekannte Typen. Erst bei der Verwendung der Klassen, Schnittstellen und Methoden werden diese Typ-Variablen durch konkrete Typen ersetzt. Damit kann typsichere Programmierung meistens gewährleistet werden. Jedoch nicht immer.[1]

Das Konzept

Ab Version 5.0 („Tiger“, 2004 veröffentlicht) steht auch in der Programmiersprache Java mit den Generics ein syntaktisches Mittel für die generische Programmierung zur Verfügung. Damit lassen sich Klassen und Methoden (Methoden auch unabhängig von ihren Klassen) mit Typen parametrisieren. Damit werden der Sprache einige ähnliche Möglichkeiten eröffnet, die sich vergleichbar bei den Templates in C++ bieten.

Prinzipiell gibt es aber durchaus wesentliche Unterschiede. Während in Java über die Schnittstelle der Typparameter parametrisiert wird, wird in C++ direkt über den Typ des Typparameters selbst parametrisiert. Der Quelltext eines C++-Templates muss für den Anwender (d. h. beim Einsetzen des Typparameters) verfügbar sein, während ein generischer Java-Typ auch als übersetzter Bytecode veröffentlicht werden kann. Für verschiedene konkret verwendete Typparameter produziert der Compiler duplizierten Zielcode.

Beispielsweise bietet die Funktion std::sort in C++ die Möglichkeit, alle Container zu sortieren, die bestimmte Methoden anbieten (hier speziell begin() und end(), die jeweils einen Iterator liefern) und deren Typparameter den 'operator<' implementiert (oder explizit eine andere Vergleichsfunktion angegeben wurde). Ein Nachteil, der sich durch dieses System ergibt, ist die (für den Programmierer!) schwierigere Übersetzung. Der Compiler hat keine andere Möglichkeit, als den Typparameter in jedem Fall durch den geforderten konkreten Typ zu ersetzen und den ganzen Code erneut zu kompilieren.

Sehr leicht können bei unpassenden Typparametern und anderen Problemen komplizierte und unverständliche Compiler-Meldungen entstehen, was einfach mit der Tatsache zusammenhängt, dass die konkreten Anforderungen an die Typparameter unbekannt sind. Die Arbeit mit C++-Templates erfordert deshalb eine lückenlose Dokumentation der Anforderungen an einen Typparameter. Durch Template-Metaprogrammierung können die meisten Anforderungen (Basisklasse, Vorhandensein von Methoden, Kopierbarkeit, Zuweisbarkeit etc.) auch in speziellen Konstrukten abgefragt werden, wodurch sich lesbarere Fehlermeldungen ergeben. Obgleich sie standardkonform sind, werden diese Konstrukte jedoch nicht von allen Compilern unterstützt.

Dagegen sind den generischen Klassen und Methoden in Java die Anforderungen (engl. constraints) an ihre eigenen Typparameter bekannt. Um eine Collection (ohne Comparator) zu sortieren, müssen die enthaltenen Elemente vom Typ Comparable sein, also dieses Interface implementiert haben. Der Compiler muss lediglich prüfen, ob der Typparameter ein Untertyp von Comparable ist, und kann damit schon sicherstellen, dass der Code korrekt ist (d. h. die erforderliche Methode compareTo verfügbar ist). Weiterhin wird ein und derselbe Code für alle konkreten Typen verwendet und nicht jedes Mal dupliziert.

Praktische Beispiele

Ein Programm verwendet eine ArrayList, um eine Liste von JButtons zu speichern.

Bisher war die ArrayList auf den Typ Object fixiert:

List list = new ArrayList();
list.add(new JButton("Button 1"));
list.add(new JButton("Button 2"));
list.add(new JButton("Button 3"));
list.add(new JButton("Button 4"));
list.add(new JButton("Button 5"));

for (int i = 0; i < list.size(); i++) {
    JButton button = (JButton) list.get(i);
    button.setBackground(Color.white);
}

Man beachte die notwendige explizite Typumwandlung (auch „Cast“ genannt) sowie die Typunsicherheit, die damit verbunden ist. Man könnte versehentlich ein Objekt in der ArrayList speichern, das keine Instanz der Klasse JButton ist. Die Information über den genauen Typ geht beim Einfügen in die Liste verloren, der Compiler kann also nicht verhindern, dass zur Laufzeit bei der expliziten Typumwandlung von JButton eine ClassCastException auftritt.

Mit generischen Typen ist in Java Folgendes möglich:

List<JButton> list = new ArrayList<JButton>();
list.add(new JButton("Button 1"));
list.add(new JButton("Button 2"));
list.add(new JButton("Button 3"));
list.add(new JButton("Button 4"));
list.add(new JButton("Button 5"));

for (int i = 0; i < list.size(); i++)
    list.get(i).setBackground(Color.white);

Beim Auslesen ist nun keine explizite Typumwandlung mehr notwendig, beim Speichern ist es nur noch möglich, JButtons in der ArrayList list abzulegen.

Ab Java7 ist die Instanzierung generischer Typen vereinfacht worden. Die erste Zeile in obigem Beispiel kann seit Java 7 folgendermaßen geschrieben werden:

List<JButton> list = new ArrayList<>();

Durch Kombination von generischen Typen mit den erweiterten For-Schleifen lässt sich obiges Beispiel kürzer fassen:

List<JButton> list = new ArrayList<>();
list.add(new JButton("Button 1"));
list.add(new JButton("Button 2"));
list.add(new JButton("Button 3"));
list.add(new JButton("Button 4"));
list.add(new JButton("Button 5"));

for (JButton b: list)
    b.setBackground(Color.white);

Ein Beispiel für eine generische Klasse, die zwei Objekte von beliebigem, aber einander gleichem Typ beinhaltet, liefert der folgende Beispielcode:

public class DoubleObject<T> {
    private T object1;
    private T object2;

    public DoubleObject(T object1, T object2) {
        this.object1 = object1;
        this.object2 = object2;
    }

    public String toString() {
        return this.object1 + ", " + this.object2;
    }

    public static void main(String[] args) {
        DoubleObject<String> s = new DoubleObject<>("abc", "def");
        DoubleObject<Integer> i = new DoubleObject<>(123, 456);
        System.out.println("DoubleObject<String> s=" + s.toString());
        System.out.println("DoubleObject<Integer> i=" + i.toString());
    }
}

Varianzfälle

In Java können die nachfolgenden Varianzfälle unterschieden werden. Sie bieten jeweils eine völlig eigenständige Flexibilität beim Umgang mit generischen Typen und sind jeweils absolut statisch typsicher.

Invarianz

Bei Invarianz ist der Typparameter eindeutig. Damit bietet Invarianz die größtmögliche Freiheit bei der Benutzung des Typparameters. Beispielsweise sind für die Elemente einer ArrayList<Integer> alle Aktionen erlaubt, die auch bei der direkten Benutzung eines einzelnen Integers erlaubt sind (inklusive Autoboxing). Beispiel:

List<Integer> list = new ArrayList<Integer>();
// ...
Integer x = list.get(index);
list.get(index).methodeVonInteger();
list.set(index, 98347); // Autoboxing, entspricht Integer.valueOf(98347)
int y = list.get(index); // Auto-Unboxing

Diese Möglichkeiten werden mit wenig Flexibilität bei der Zuweisung von Objekten der generischen Klasse selbst erkauft. Beispielsweise ist Folgendes nicht erlaubt:

List<Number> list = new ArrayList<Integer>();

und das, obwohl Integer von Number abgeleitet ist. Der Grund liegt darin, dass der Compiler hier nicht mehr sicherstellen kann, dass keine Typfehler auftreten. Mit Arrays, die eine solche Zuweisung erlauben, hat man schlechte Erfahrungen gemacht:

// OK, Integer[] ist abgeleitet von Number[]
Number[] array = new Integer[10];

// ArrayStoreException zur Laufzeit: Double -> Integer sind nicht
// zuweisungskompatibel
array[0] = new Double(5.0);

Kovarianz

Man bezeichnet Arrays als kovariant, was besagt:

Aus T extends V folgt: T[] extends V[]

oder allgemeiner:

Aus T extends V folgt: GenerischerTyp<T> extends GenerischerTyp<V>

Es verhält sich also der Array-Typ bzgl. der Vererbungshierarchie genauso wie der Typparameter. Kovarianz ist auch mit generischen Typen möglich, allerdings nur mit Einschränkungen, so dass Typfehler zur Kompilierzeit ausgeschlossen werden können.

Referenzen müssen mit der Syntax ? extends T explizit als kovariant gekennzeichnet werden. T heißt upper typebound, also der allgemeinste Typparameter, der erlaubt ist.

List<? extends Number> list;
list = new ArrayList<Double>();
list = new ArrayList<Long>();
list = new ArrayList<Integer>();

// Typfehler vom Compiler
list.set(index, myInteger);

// OK aber Warnung vom Compiler: unchecked cast
((List<Integer>) list).set(index, myInteger);

Das Ablegen von Elementen in diesen Listen ist nicht möglich, da dies, wie oben beschrieben, nicht typsicher ist (Ausnahme: null kann abgelegt werden). Bereits zur Kompilierzeit tritt ein Fehler auf. Allgemeiner gesagt, ist die Zuweisung

?? extends T

nicht erlaubt.

Möglich dagegen ist das Auslesen von Elementen:

Number n = list.get(index); // OK
Integer i = list.get(index); // Typfehler: Es muss sich bei '? extends Number'
                             // nicht um ein Integer handeln.
Integer j = (Integer) list.get(index); // OK

Die Zuweisung

? extends TT (oder Basisklasse)

ist also erlaubt, nicht aber die Zuweisung

? extends Tabgeleitet von T

Generics bieten also wie Arrays kovariantes Verhalten, verbieten aber alle Operationen, die typunsicher sind.

Kontravarianz

Kontravarianz bezeichnet das Verhalten der Vererbungshierarchie des generischen Typs entgegen der Hierarchie seines Typparameters. Übertragen auf das obige Beispiel würde das bedeuten: Eine Liste<Number> wäre zuweisungskompatibel zu einer Liste<Double>. Dies wird folgendermaßen bewerkstelligt:

List<? super Double> list;
list = new ArrayList<Number>();
list = new ArrayList<Double>();
list = new ArrayList<Object>();

Ein Objekt, das sich kontravariant verhält, darf keine Annahmen darüber machen, inwiefern ein Element vom Typ V von T abgeleitet ist, wobei T der lower Typebound ist (im Beispiel von ? super Double ist T Double). Deshalb kann aus den obigen Listen nicht gelesen werden:

// Fehler: 'list' könnte vom Typ List<Object> sein
Number x = list.get(index);

// Fehler: 'list' könnte List<Object> oder List<Number> sein
Double x = list.get(index);

// Die einzige Ausnahme: Objects sind auf jeden Fall in der Liste
Object x = list.get(index);

Nicht erlaubt, da nicht typsicher, ist also die Zuweisung ? super T → (abgeleitet von Object)

Unschwer zu erraten: Im Gegenzug kann in eine solche Liste ein Element abgelegt werden:

List<? super Number> list;
list.add(new Double(3.0)); // OK: 'list' hat immer den Typ List<Number>
                           // oder List<Basisklasse von Number>. Damit
                           // ist die Zuweisung Double -> T immer erlaubt.

Uneingeschränkte parametrische Polymorphie

Zu guter Letzt bieten Generics noch gänzlich polymorphes Verhalten an. Hierbei kann keinerlei Aussage über die Typparameter gemacht werden, denn es wird in beide Richtungen keine Grenze angegeben. Dafür wurde die Wildcard definiert. Sie wird durch ein Fragezeichen repräsentiert.

List<?> list;
list = new ArrayList<Integer>();
list = new ArrayList<Object>();
list = new ArrayList<String>();
// ...

Der Typparameter selbst kann hierbei nicht genutzt werden, da keine Aussage möglich ist. Lediglich die Zuweisung T → Object ist erlaubt, da T auf jeden Fall ein Object ist. Im Gegenzug ist garantiert, dass der Code mit allen Ts arbeiten kann.

Nützlich kann so etwas sein, wenn man nur mit dem generischen Typ arbeitet:

// Keine Informationen über den Typparameter nötig, kann ''beliebige'' Listen
// aufnehmen.
int readSize(List<?> list) {
    return list.size();
}

Zur Verdeutlichung, dass hier Wildcards unnötig sind, und es eigentlich gar nicht um irgendeine Varianz geht, sei folgende Implementierung der obigen Funktion angegeben:

<T> int readSize(List<T> list) {
    return list.size();
}

Einzelnachweise

  1. Java and Scala’s Type Systems are Unsound.