Collections Sind Subklassen-Objekte in Listen mit Generics erlaubt?

Thor_es

Mitglied
Hallo zusammen,

bisher dachte ich:
  • dass einer typisierten Liste nur Objekte des angegeben Typs hinzugefügt werden können
  • dass ich abgeleitete Objekte upcasten muss, um sie der auf die Superklasse typisierten Liste hinzufügen zu können
  • beim Abruf aus der Liste nur Objekte des angegebenen Listentyps zurückgegeben werden (abgeleitete Objekte werden als Superklassentyp ausgegeben)
Nun habe ich folgendes Beispiel erstellt. Es gibt eine Superklasse "Animal" von der die Unterklassen "Cat" und "Dog" ableiten. Die Liste Tiere ist auf "Animal" typisiert. Ich kann nun alle Unterklassen-Objekte ohne Upcast hinzufügen (von ich dachte, dass es nicht funktioniert). Außerdem werden die "Cat"- und "Dog"-Objekte auch als diese aus der als "Animal" typisierten Liste zurückgegeben und die korrekten Methoden werden ausgeführt.

Das widerspricht meinem aktuellen Verständnis von Generics. Kann mir bitte jemand helfen, meinen Knoten im Kopf zu lösen.

Vielen Dank für eure Unterstützung.

[CODE lang="java" title="Beispielcode" highlight="10, 11, 12, 13, 14, 15"]import java.util.*;

public class Main
{
public static void main(String[] args) {
List<Animal> Tiere = new ArrayList<>();
Tiere.add(new Cat());
Tiere.add(new Dog());
Tiere.add(new Animal());
System.out.println(Tiere.get(0).getClass()); //Output: class Cat
System.out.println(Tiere.get(1).getClass()); //Output: class Dog
System.out.println(Tiere.get(2).getClass()); //Output: class Animal
Tiere.get(0).sound(); //Output: Miau
Tiere.get(1).sound(); //Output: Wuff
Tiere.get(2).sound(); //Output: GrrrrMiauWuffPiep
}
}

class Animal{
public void sound(){
System.out.println("GrrrrMiauWuffPiep");
}
}
class Cat extends Animal{
@Override
public void sound(){
System.out.println("Miau");
}
}
class Dog extends Animal{
@Override
public void sound(){
System.out.println("Wuff");
}
}[/CODE]
 

Robert Zenz

Top Contributor
dass einer typisierten Liste nur Objekte des angegeben Typs hinzugefügt werden können
Das ist auch richtig, aber nehmen wir folgende Kette an:

Java:
public class A extends Object {}
public class B extends A {}
public class C extends C {}

public class X extends Object {}
public class Y extends X {}
public class Z extends Z {}

Dann sind *alle* diese Klassen ja ein Object, aber sie sind halt auch mehr. Wenn du diese jetzt zuweist:

Java:
C c = new C();

Object object = c;

// TODO Use "object" here.

Dann funktioniert dies ja auch ohne Probleme, weil sie eben von diesem Typ erben.

dass ich abgeleitete Objekte upcasten muss, um sie der auf die Superklasse typisierten Liste hinzufügen zu können
Nein.

beim Abruf aus der Liste nur Objekte des angegebenen Listentyps zurückgegeben werden (abgeleitete Objekte werden als Superklassentyp ausgegeben)
Ja, wenn es um den Compiler geht. Klassen von vorhin:

Java:
C c = new C();

List<A> listOfAs = new ArrayList<>();
listOfAs.add(c); // Keine Umwandlung notwendig, weil "C" ja auch "A" ist.

A retrievedC = litOfAs.get(0);

Du kannst natuerlich ein Objekt vom Typ C in die Liste packen, weil diese ja auch vom Typ [/ICODE]A[/ICODE]. Aber aus der Liste bekommst du nur den spezifizierten Typen heraus, weil der Compiler kann ja nicht wissen von welchen Subtypen die Werte sind, der Compiler kann nur garantieren dass diese vom Typ [/ICODE]A[/ICODE] sind.

Oder um es etwas anders zu veranschaulichen:

Java:
new A() instanceof A // true
new B() instanceof A // true
new C() instanceof A // true

new A() instanceof B // false
new B() instanceof B // true
new C() instanceof B // true

new A() instanceof C // false
new B() instanceof C // false
new C() instanceof C // true
 

Thor_es

Mitglied
Ja, wenn es um den Compiler geht. Klassen von vorhin:

Java:
C c = new C();

List<A> listOfAs = new ArrayList<>();
listOfAs.add(c); // Keine Umwandlung notwendig, weil "C" ja auch "A" ist.

A retrievedC = litOfAs.get(0);

Du kannst natuerlich ein Objekt vom Typ C in die Liste packen, weil diese ja auch vom Typ [/ICODE]A[/ICODE]. Aber aus der Liste bekommst du nur den spezifizierten Typen heraus, weil der Compiler kann ja nicht wissen von welchen Subtypen die Werte sind, der Compiler kann nur garantieren dass diese vom Typ [/ICODE]A[/ICODE] sind.

Daher bin ich auch davon ausgegangen, dass der Compiler, nachdem ich die Objekte aus der Liste hole, nicht mehr ermitteln kann, um welches Animal es sich genau handelt und ich über instanceof prüfen müsste, ob es sich um ein Objekt "Cat" oder "Dog" handelt. In meinem Codebeispiel konnte ich aber die Objekte ohne Aufwand als Typ Cat und Dog verwenden, obwohl die Liste als Animal typisiert ist. Kannst du mir das noch einmal erklären?
 

Robert Zenz

Top Contributor
Daher bin ich auch davon ausgegangen, dass der Compiler, nachdem ich die Objekte aus der Liste hole, nicht mehr ermitteln kann, um welches Animal es sich genau handelt und ich über instanceof prüfen müsste, ob es sich um ein Objekt "Cat" oder "Dog" handelt.
Der Compiler kann das auch nicht mehr, aber zur Laufzeit ist die Instanz ja natuerlich weiterhin so wie sie erzeugt wurde. Als beim Aufruf:

Java:
C c = (C)listOfAs.get(0);

Kann dir der Compiler nicht garantieren dass das wirklich ein "C" ist welches du holst, aber zur Laufzeit ist die Instanz natuerlich so wie sie ist.

In meinem Codebeispiel konnte ich aber die Objekte ohne Aufwand als Typ Cat und Dog verwenden, obwohl die Liste als Animal typisiert ist.
Weil sich die *Instanz* nicht aendert, nur weil du sie irgendwohin castest. Das aendert weder die Instanz noch den Typ von der Instanz.

Wenn du eine Katze hast, und diese in einem Raum mit *anderen* Tieren steckst, bekommst ja auch wieder eine Katze heraus, und nicht *irgendein Tier*.
 
M

Mart

Gast
wenn du einen "cast" anwendest sagst du dem compiler eigentlich " trust me im an engineer" ... du sagst halt vertrau mir mit dem cast und behandle es wie wenn das so eines wäre ,,die Urpsrungliche Klasse von der ein Objekt herkommt darf nicht verändert werden
 
M

Mart

Gast
Das ist eine falsche Annahme, du verwendest die Instanzen *als* Animal Instanzen, aber es sind Instanzen von den Typen Cat und Dog.
da du weist dass jedes animal eine sound methode hat kannst du egal was du von dem vererbungs baum bekommst die methode aufrufen ... was in der methode drin steht ist das problem des objektes ( bzw so wies in der jeweiligen klasse definiert worden ist )
 

Thor_es

Mitglied
OK. Wenn ich nun aber die "sound"-Methode aus der Oberklasse "Animal" entferne, erhalte ich beim Aufruf der entsprechenden Methoden bei "Cat" und "Dog" die folgende Fehlermeldung:

Screenshot.png

Dann muss ich wieder umständlich downcasten (siehe Codebeispiel), damit es funktioniert, obwohl die ursprünglichen Typen der Instanzen bekannt sind und auch die Methode getClass() die korrekten Subklassen ausgibt. Das kann ich noch nicht ganz nachvollziehen.

[CODE lang="java" title="geänderter Code" highlight="27, 14, 15, 17-22"]import java.util.*;

public class Main
{
public static void main(String[] args) {
List<Animal> Tiere = new ArrayList<>();
Tiere.add(new Cat());
Tiere.add(new Dog());
Tiere.add(new Animal());
System.out.println(Tiere.get(0).getClass()); //Output: class Cat
System.out.println(Tiere.get(1).getClass()); //Output: class Dog
System.out.println(Tiere.get(2).getClass()); //Output: class Animal

Tiere.get(0).sound(); //Output: error
Tiere.get(1).sound(); //Output: error

Animal tier1 = Tiere.get(0);
Cat katze = (Cat)tier1;
katze.sound();
Animal tier2 = Tiere.get(1);
Dog hund = (Dog)tier2;
hund.sound();
}
}

class Animal{

}
class Cat extends Animal{
public void sound(){
System.out.println("Miau");
}
}
class Dog extends Animal{
public void sound(){
System.out.println("Wuff");
}
}[/CODE]
 
M

Mart

Gast
das objekt wird behandelt als irgendwas beim vererbungs baum beginnnend ab animal, durch die generic Liste sagst du "egal was ich rein lege es is MINDESTENS ein ANimal"

dh wenn du get machst kriegst du irgendetwas das MINDESTENS ein Animal ist raus, das ist das einzige was die jvm zu dem zeitpunkt sicher weis
das objekt wird als ein Animal behandelt ( dh alle methodne die public und package sind von außen ) können aufgerufen werden aber der Inhalt dieser Methodne ist das problem des objektes wie bereits gesagt :)

wenn du im vererbungs baum beim animal eine public methode hast MÜSSEN alle unteren Klassen in der Vererbung diese Public methode auch haben dh jetzt kombinieren , "es ist mindestens ein animal und jedes animal kann dieses und jenes und alle drunter können das auch" deweswegen durftest du auch wei du sound noch in animal hattest die methode aufrufen durftest , der Inhalt der methode ist wie oben beschrieben das problem der Klasse/objektes

downcasts sind wie besagt "trust me im an engineer" für den Compiler... der Compiler vertraut dir da einfach mal weil der compiler nicht weis ob es funtkionieren wird wie erwartet, falls es wie erwartet funktioniert hast du am ende ein objekt das momentan als der cast gedeutet wird deswegen darfst du dann sound aufrufen weils dann als cat zb gedeutet wird

kurzgesagt... du deutest nach dem Get das object als animal aber ein animal kann keinen sound mehr weil das eine Spezielle methode IRGENDEINER unterklasse ist ... zu was soll den gecastet werden ? hund katze? ausprobieren ? zufall ? -> uneindeutigkeit darfs nicht geben
 

Thor_es

Mitglied
Klasse. Vielen Dank für die ausführliche Erklärung. So langsam erschließt es sich mir 😅

Nur noch eine kleine Rückfrage bezüglich des Cast-Syntax. Gibt es hier eine "schlanke" Variante direkt aus der Liste auf "Cat" oder "Dog" zu casten. Mit "Cat Katze = (Cat)Tiere.get(0)" hatte ich keinen Erfolg.
 
Ähnliche Java Themen
  Titel Forum Antworten Datum
K Warum sind Werte in den Feldern ? Java Basics - Anfänger-Themen 2
T Methode, die prüft ob in einem Int-Array maximal 2 Zahlen enthalten sind, die größer als ihr Vorgänger sind Java Basics - Anfänger-Themen 5
berserkerdq2 Wie würde man einen regulären Ausdruck in Java schreiben, der prüft, dass zwei bestimtme Zahlen nicht nebeneinadner sind? Java Basics - Anfänger-Themen 3
milan123 das ist meine aufgabe ich hab das problem das bei mir Wenn ich die Richtung der Linien verändern will und drei davon sind richtig, verändere ich die 4 Java Basics - Anfänger-Themen 3
Igig1 Welche Werte sind als default Werte in einem Array, der als Datentyp eine Klasse hat? Java Basics - Anfänger-Themen 1
X Wie erreiche ich, dass ein Robot weitere Attribute hat, die nicht materialisiert sind, sondern nur über get/ set-Methoden simuliert sind? Java Basics - Anfänger-Themen 1
M Wie können Klassen nicht-materialisierte Attribute haben, die nur über get/ set-Mehoden simuliert sind? Java Basics - Anfänger-Themen 6
C Sind die while-Schleifen richtig in for-Schleifen ersetzt worden? Java Basics - Anfänger-Themen 8
S Sind unten stehende Anweisungen kompilierbar? Java Basics - Anfänger-Themen 7
M Wie kann ich Werte die in einer While Schleife sind weiter genutzt werden? Java Basics - Anfänger-Themen 7
CptK Generics: Klassen die Interface implementieren, aber selbst nicht das Interface sind Java Basics - Anfänger-Themen 8
mhmt_03 dafür sorgen, dass im JTextfield nur zahlen eingebbar sind Java Basics - Anfänger-Themen 9
M Warum werden character, die Leerzeichen sind, nicht korrekt verarbeitet? Java Basics - Anfänger-Themen 2
M Scannen von *.txt - Dateien; wo sind der oder die Fehler? Java Basics - Anfänger-Themen 4
L Methode implementieren, Parameter die übergeben werden sind final Java Basics - Anfänger-Themen 4
S Laufzeit Quicksort wenn alle Elemente gleich sind Java Basics - Anfänger-Themen 4
B Sind meine If-Statements richtig angesetzt ? Java Basics - Anfänger-Themen 27
A Haben KNNs ein Gedächtnis, lernen etwas oder verändern sich, während sie nicht trainieren, aber aktiv sind? Java Basics - Anfänger-Themen 3
M prüfen ob alle array werte gleich sind Java Basics - Anfänger-Themen 27
R Wozu sind Annotations da? Java Basics - Anfänger-Themen 3
H Was sind Package bei eclipse? Java Basics - Anfänger-Themen 1
V NullPointerException, wenn Key und Value null sind Java Basics - Anfänger-Themen 2
G Überprüfen ob alle Ziffern von 1-9 in einem Integer vorhanden sind Java Basics - Anfänger-Themen 6
D Zwei Strings sind gleich bei if aber nicht true Java Basics - Anfänger-Themen 2
J Strings sind gleich werden aber ungleich ausgewertet Java Basics - Anfänger-Themen 2
H Array mit Zahlen die durch 3 und 5 teilbar sind erstellen Java Basics - Anfänger-Themen 13
J Klassen Math && Random: wie zufällig sind Zufallszahlen? Java Basics - Anfänger-Themen 19
L Prüfe, ob die im String Array enthaltenen Strings aufsteigend sind. Java Basics - Anfänger-Themen 19
D Fehlermeldung obwohl Variablen bereits deklariert sind? Java Basics - Anfänger-Themen 14
C Tabs in JTabbedPane wechseln, wenn Tabs in eigenen Klassen sind Java Basics - Anfänger-Themen 2
Azazel Wie wichtig sind Castings in Java ? Java Basics - Anfänger-Themen 1
S Was sind Java Beans? Java Basics - Anfänger-Themen 7
S Erste Schritte Generische Klassen sind toll ....aber warum sollte ich das je benutzen? Java Basics - Anfänger-Themen 3
J Prüfen ob Arrays nur mit einem Wert belegt sind Java Basics - Anfänger-Themen 3
K Erste Schritte switch - Warum sind long/float/double/... nicht erlaubt? Java Basics - Anfänger-Themen 5
M Wie sicher sind Daten im Java Programm? Java Basics - Anfänger-Themen 9
T Wie vergleiche ich die Jahre aus der while Schleife die in ( public class) fuer cbx geschrieben sind Java Basics - Anfänger-Themen 5
P Wieviele Tage seit dem Datum vergangen sind Java Basics - Anfänger-Themen 5
P OOP Testen ob 2 Strings gleich sind Java Basics - Anfänger-Themen 8
M Welche externen Bibliotheken sind in Java sehr zu empfehlen? Java Basics - Anfänger-Themen 4
? Wie sind ESCAPE-Sequenzen (z.B \f für einen Seitenvorschub) richtig anuwenden? Java Basics - Anfänger-Themen 3
M Warum sind Strings Immutable? Java Basics - Anfänger-Themen 7
S Werte aus SingeltonKlasse sind manchmal =0 &manchmal !=0 Java Basics - Anfänger-Themen 1
M Sind solche boolean Anweisen empfehlenswert? Java Basics - Anfänger-Themen 3
F Scanner + Stringbuilder geben leeren String aus wenn Umlaute enthalten sind Java Basics - Anfänger-Themen 29
M String überprüfen ob nur Buchstaben enthalten sind? Java Basics - Anfänger-Themen 10
Kenan89 Wo sind die Java Standard Library Source Codes zu finden? Java Basics - Anfänger-Themen 5
L JDK installieren Sind in src.zip tatsächlich die verwendeten Klassen? Java Basics - Anfänger-Themen 7
L Byte[] to String, doch bits sind gespiegelt (MSB/LSB) Java Basics - Anfänger-Themen 3
B Funktionen programmieren, die im Hintergrund aktiv sind Java Basics - Anfänger-Themen 2
S Von byte[] nach String zurueck nach byte[]. Arrays sind nicht identisch :( Java Basics - Anfänger-Themen 6
C hashCode() bei Klassen, die nicht immutable sind Java Basics - Anfänger-Themen 27
C Erste Schritte felder, die public final sind Java Basics - Anfänger-Themen 6
D Warum sind Generics mit Vorsicht zu genießen? Java Basics - Anfänger-Themen 6
E Was sind Javascript und Java EE? Java Basics - Anfänger-Themen 7
S Wie performance lastig sind rekursionen Java Basics - Anfänger-Themen 13
C Nach Java-Installation sind Befehle erfolglos Java Basics - Anfänger-Themen 4
B Variablen Warum sind die blau Java Basics - Anfänger-Themen 2
L Liste aller Klassen die in einem Paket sind Java Basics - Anfänger-Themen 7
S Warten bis alle Threads fertig sind Java Basics - Anfänger-Themen 12
M Erste Schritte zwei Buchstaben die im String enthalten sind ausgeben Java Basics - Anfänger-Themen 21
J Drei Errors sind drei zuviel! Java Basics - Anfänger-Themen 25
RySa Input/Output Datei kann nicht gelöscht werden, obwohl Streams geschlossen sind. Java Basics - Anfänger-Themen 2
H Wieviele Objekte gleichzeitig sind sinnvoll? Java Basics - Anfänger-Themen 4
S Dezimale Konstanten sind immer positiv oder null - was heisst das den genau? Java Basics - Anfänger-Themen 2
D Strings sind ungleich obwohl sie in der Ausgabe gleich sind Java Basics - Anfänger-Themen 10
D Sind Enums typsichere Konstanten? Java Basics - Anfänger-Themen 15
S Warum sind Attribute der Klasse java.awt.Point public? Java Basics - Anfänger-Themen 3
T Buttons (auf denen bilder sind) random vertauschen Java Basics - Anfänger-Themen 11
W Array nach Elemenden die durch 2 teilbar sind durchsehen Java Basics - Anfänger-Themen 9
N TextZeile in einzelne Strings teilen, die mit Komma getrennt sind Java Basics - Anfänger-Themen 4
L Elemente die in Array1 sind aus Array2 löschen Java Basics - Anfänger-Themen 6
C Compiler-Fehler Variablen sind angeblich nicht initialisiert Java Basics - Anfänger-Themen 7
M Sind ternäre Operatoren für einen guten Programmierstil wichtig ? Java Basics - Anfänger-Themen 10
I Was sind denn überhaupt Doctypes bei Java? Java Basics - Anfänger-Themen 7
Developer_X Sounds abspielen, die im internet sind (.wav) Java Basics - Anfänger-Themen 2
A Erzeugte Objekte sind nicht bekannt Java Basics - Anfänger-Themen 16
H Warum sind in Java Strings und Arrays eigentlich unveränderlich? Java Basics - Anfänger-Themen 2
G Objekte von Klassen die erst zur Laufzeit bekannt sind erstellen Java Basics - Anfänger-Themen 6
nrg Datentypen Sind Arrays Objekte? Java Basics - Anfänger-Themen 9
B Für was sind die Annotationen gut? Java Basics - Anfänger-Themen 6
F import Statement für Klassen die dem default Package zugeordnet sind? Java Basics - Anfänger-Themen 2
T Probleme mit ArrayList, in der Objekte gespeichert sind, die ArrayLists enthalten Java Basics - Anfänger-Themen 2
Semox Sind Objekte in einer ArrayList überschreibbar? Java Basics - Anfänger-Themen 2
D 100.0% gleichmäßig aufteilen, so dass Summe 100.0% sind, nicht 99,9% oder 100,1% Java Basics - Anfänger-Themen 3
feuervogel Datentypen Zwei Sets sind gleich, aber dann doch nicht Java Basics - Anfänger-Themen 9
N java.util.Arrays.sort Warum sind Leerzeichen vor alphabetischen Zeichen sortiert? Java Basics - Anfänger-Themen 12
cowabunga1984 Unit-Testing - Welche Testfälle sind relevant? Java Basics - Anfänger-Themen 4
M Was sind das für Probleme? Java Basics - Anfänger-Themen 6
B was sind Dataflavor Java Basics - Anfänger-Themen 2
M Wo sind meine double-Zahlen? Java Basics - Anfänger-Themen 2
T Sind Kontrollflussanweisungen auch Methoden? Java Basics - Anfänger-Themen 5
N Wie schlimm sind Cyclen in einem Programm Java Basics - Anfänger-Themen 3
G welche Teile der api sind wichtig? Java Basics - Anfänger-Themen 3
T Was sind Attribute? Java Basics - Anfänger-Themen 9
P Was sind Vectoren und Maps in Java? Java Basics - Anfänger-Themen 3
G Sind das Generics Java Basics - Anfänger-Themen 2
P Prüfen ob alle zellen der tabelle leer sind Java Basics - Anfänger-Themen 9
C Alle Zeichen, die in der Console sind in Textdatei Java Basics - Anfänger-Themen 12
M Feststellen, das Textboxen leer sind! Java Basics - Anfänger-Themen 3

Ähnliche Java Themen

Neue Themen


Oben