Frage zu Interfaces (Beispiel: Comparable)

Status
Nicht offen für weitere Antworten.

Marcel_Handball

Bekanntes Mitglied
Hi,
ich habe zwei Verständisfragen zu Interfaces.

Wieso ist eine Konstruktion wie die folgende nicht möglich?

Java:
Comparable<String>[] c = {"eins","zwei"};
bzw.: Comparable<String>[] c = new Comparable<String>[5];


Andererseits funktioniert folgendes:

Java:
Comparable[] objects = new Comparable[2];
objects[0] = "String";
objects[1] = "String2";

System.out.println( (String) getSmallest(objects) );

--------------

public static Object getSmallest(Comparable[] objects){
     Object smallest = objects[0];
     for(int i = 1; i < objects.length; i++){
          if(objects[i].compareTo(objects[i+1]) > 0){
               smallest = objects[i];
          }
     }
}

Wieso ist es möglich Objekte vom Typ Comparable<String> einem Comparable-Objekt zuzuweisen, sodass anschließend in der Methode getSmallest() auf das ursprüngliche Comparable<String>-Objekt die Methoden compareTo(Object) aufrufen kann, obwohl die Klasse String nur compareTo(String) definiert?
 
S

SlaterB

Gast
1.
Arrays kann man generell nicht generisch erzeugen, da ist eine Barriere in Java
> new Comparable<String>[5];
gibts also nicht,

> Comparable<String>[] c = {"eins","zwei"};
könnte dagegen funktionieren, kann es grad nicht testen, was ist denn die Fehlermeldung?

-----

2.
das Interface Comparable benutzt nur einen generischen Parameter T, die Methode lautet
compareTo(T o)

zur Laufzeit existierten in Java die generischen Informationen weitgehend nicht mehr, alle Ts werden zu Object,

edit:
ich glaube mir gerade selber gar nicht mehr, muss ich doch noch mal nachschauen ;) einen Moment

edit2:
nun mit Java-Zugang:
Java:
public class Test extends Random {

	public static void main(String[] args) {

		Comparable a = 4;
		Comparable b = "";
		System.out.println(a.compareTo(b));
	}
}
die Methoden sind doch eingeschränkt:
Code:
Exception in thread "main" java.lang.ClassCastException: java.lang.String cannot be cast to java.lang.Integer
	at java.lang.Integer.compareTo(Integer.java:35)
	at testmain.Test.main(Test.java:16)
bei String zu String klappte es dank gleichen Typ

hier trifft man also auf das 'weitgehend' aus 'zur Laufzeit existierten in Java die generischen Informationen weitgehend nicht mehr, alle Ts werden zu Object,'
bzw. das Problem ist gar nicht mehr generisch, die Methode ist fest implementiert, nur erfüllt diese feste Implementierung die generische Anforderung,
tja, ganz schön wirr, sorry ;)

Java:
public class Test {

	public static void main(String[] args) {
		Comparable a = new ComparableTest1<String>();
		a.compareTo(4); // geht

		Comparable b = new ComparableTest2<List>();
		b.compareTo(4); // ClassCastException
	}
}

class ComparableTest1<T> implements Comparable<T> {

	public int compareTo(T o) {
		return 0;
	}
}

class ComparableTest2<T> implements Comparable<String> {

	public int compareTo(String o) {
		return 0;
	}
}
 
Zuletzt bearbeitet von einem Moderator:

Marcel_Handball

Bekanntes Mitglied
Hey SlaterB,

erst einmal danke für deine ausführliche Antwort. Ich hab mir deinen Code sicherlich jetzt 5min angeschaut und überlegt, wieso dies so gilt.
Da ich gerade erst angefangen habe, meine Java-Kenntnisse wieder aufzuarbeiten, bin ich zu Generics nicht auf der Höhe.

Es ist also möglich einem "unbestimmten generischen" Objekt einem bestimmtes zuzuordnen:

Java:
Comparable a = new ComparableTest1<String>();

Doch setzt der Compiler dadurch, wenn ich dich richtig verstanden habe, die Methode compareTo(T o) von ComparableTest1 nicht auf compareTo(String o). (Sondern auf Object?)

Wie verhält es sich wenn ich das Object ohne Typangabe initialisiere?

Java:
Comparable a = new ComparableTest1();

Wie sieht dann die Methode intern aus?


Würde die Zeile: a.compareTo(4); auch funktionieren, wenn man den Vergleich tatsächlich vornehmen würde (denn dann würde man doch versuchen auf Eigenschaften von T zuzugreifen, die das übergebenen Objekt unter Umständen gar nicht besitzt, da man den Typ des Objekts nicht kennt)
 
S

SlaterB

Gast
> Doch setzt der Compiler dadurch, wenn ich dich richtig verstanden habe, die Methode compareTo(T o) von ComparableTest1 nicht auf compareTo(String o). (Sondern auf Object?)

new ComparableTest1<String>(); hat nur eine Methode compareTo(Object), ja,
das liegt an der Definition dieser Klasse, nicht an dem Speichern in der Comparable-Variable, wie dein Satzbau vermuten läßt

> Wie verhält es sich wenn ich das Object ohne Typangabe initialisiere?

intern zur Laufzeit ist es in beiden Fällen Object,
ohne Typangabe wird auch der Compiler von Object ausgehen

> Würde die Zeile: a.compareTo(4); auch funktionieren, wenn man den Vergleich tatsächlich vornehmen würde (denn dann würde man doch versuchen auf Eigenschaften von T zuzugreifen, die das übergebenen Objekt unter Umständen gar nicht besitzt, da man den Typ des Objekts nicht kennt)

für a, also ComparableTest1, ist im Quellcode der Klasse der Typ ja gar nicht bekannt, daher kann auf keine spezielle Eigenschaft zugegriffen werden,

anders sieht es in ComparableTest2 aus, da wird dann mit String gearbeitet, man könnte String-Methoden aufrufen, ClassCastException ist möglich,
bzw. erfolgt ja jetzt schon durch den Cast auf den Parameter-Typ
 

Marcel_Handball

Bekanntes Mitglied
Heißt das, dass eine Objektvariable wie [T o], [Object o] und eine Funktion wie compareTo(T o), compareTo(Object o) entspricht?

Wenn alle T's zu Objects umgewandelt werden, wozu gibt man einem Objekt bei seiner Initialisierung einen Generic mit?
Welcher Unterschied besteht zwischen einem direkten Initisilisierung von compareTo(Object) und compareTo(T)?
 
S

SlaterB

Gast
nur der Compiler beachtet T statt Object und wird meckern, falls man nicht den richtigen Typ übergibt,
eine Hilfe zur Bearbeitung des Quellcodes, für die Programmausführung weitgehend unerheblich und auch in der Programmierung durch das Casten teilweise zu verschleiern:

Comparable a = new ComparableTest1<String>();
oder
Comparable a = new ComparableTest1<MondRakete>();
macht niemals nirgendwo im Programm einen Unterschied, der TypParameter ist völlig nutzlos

------

Comparable<String> a = new ComparableTest1<String>();
oder
Comparable<MondRakete> a = new ComparableTest1<MondRakete>();
bringt im weiteren Quellcode den Vorteil, dass der Compiler anmerken kann, dass ein compareTo-Aufruf mit Typ Integer eher sinnlos ist,
zur Lauftzeit wird daraus aber in allen Fällen:
Comparable a = new ComparableTest1(); mit compareTo(Object)

nicht so bei ComparableTest2 oder der Klasse String, deinem ursprünglichen Beispiel, da ist compareTo(String) fest implementiert, relativ unabhängig von Generics
 

Marcel_Handball

Bekanntes Mitglied
Alles klar, also ist T immer vom Typ Object und nur zur Fehlerprevention beim Programmieren gedacht (Anmerkender Compiler).

Man kann also nicht davon ausgehen, dass der Parameter von compareTo vom Typ des Generics (des Objektes) ist, sodass eine Überprüfung wie (this instanceof paramter) nötig ist für einen fehlerlosen Vergleich.
 
S

SlaterB

Gast
wenn der Parameter generisch ist, dann ist auch keine instanceof-Prüfung möglich

überhaupt ist auch ein Vergleich, ob fehlerlos oder nicht, quasi ausgeschlossen, falls man nicht Methoden von Object wie hashCode() heranzieht,
viel kann man mit generischen Parametern nicht machen,
am besten sind da noch die Collections, Objekte einlagern und wieder zurückgeben, ohne sie sich genau anzuschauen


interessant wird es noch, wenn man die Basisklasse des generischen Parameter festlegt:

Java:
public class Test {

	public static void main(String[] args) {
		Comparable a = new ComparableTest1<Double>();
		a.compareTo(Integer.valueOf(4)); // geht, Integer ist Number
		a.compareTo(""); // ClassCastException
	}
}

class ComparableTest1<T extends Number> implements Comparable<T> {

	public int compareTo(T o) {
		return o.shortValue(); // Number-Methoden können aufgerufen werden
	}
}
hier wird im Programm T nicht zu Object, sondern zu Number, dann können auch Number-Methoden aufgerufen werden
 

Marcel_Handball

Bekanntes Mitglied
Die Klasse String implementiert Comparable<String>, muss also die compareTo-Methode überschreiben (aufgrund des Generics mit einem String-Parameter).
Ein Aufruf der Methode mit einem Object-Objekt ist nicht möglich.

Also wird T immer nur durch Object ersetzt wenn keine Angabe zum Typ von T gemacht wird?

Der Versuch instanceof auf einen Generic anzuwenden funktioniert allerdings bei mir.

Java:
class ComparableTest1<T> implements Comparable<T> {
 
    public int compareTo(T o) {
        System.out.println(o instanceof Integer);
        System.out.println((Integer) o);
    	return 0;
    }
}
 
Zuletzt bearbeitet:
S

SlaterB

Gast
> Also wird T immer nur durch Object ersetzt wenn keine Angabe zum Typ von T gemacht wird?

so in etwa

> Der Versuch instanceof auf einen Generic anzuwenden funktioniert allerdings bei mir.

nun gut, das geht,
aber bei allgemeinem Parameter T ist es recht sinnfrei, auf Integer zu prüfen,

noch was denkbares ist, am Anfang im Konstruktor ein Class<T>-Objekt zu übergeben,
dann könnte man später
class.isInstance(o)
testen

ComparableTest1<String> x = new ComparableTest1<String>(String.class) // Integer.class ginge nicht, passt nicht zum generischen Parameter
 
S

SlaterB

Gast
in jedem Windows-JDK-Verzeichnis sollte eine Datei src.zip dabei sein (unter Ubuntu Extra-Paket)
 
Status
Nicht offen für weitere Antworten.
Ähnliche Java Themen
  Titel Forum Antworten Datum
rentasad Design-Frage - Interfaces, Klassen, statische Methoden Allgemeine Java-Themen 3
T Nochmal Frage zu Vererbung Interfaces etc. Allgemeine Java-Themen 10
KonradN Mal eine Frage zu Binary Serialization Allgemeine Java-Themen 15
8u3631984 Frage zu Java Streams min / max Allgemeine Java-Themen 17
8u3631984 Frage Performance bei Linked List und Array List Allgemeine Java-Themen 5
H Frage regex greater than less than Allgemeine Java-Themen 7
berserkerdq2 Frage zu IntelliJ und JavaFX Allgemeine Java-Themen 1
W Timer Konzept-Frage Allgemeine Java-Themen 16
T Eine Frage des Designs Allgemeine Java-Themen 2
C Frage zu eigenem TableCellRenderer Allgemeine Java-Themen 11
C Programmvorstellung & Frage zum Thema Geschäftsform Allgemeine Java-Themen 51
J Frage zu System.getproperties. Allgemeine Java-Themen 60
molat100 wie kann man die Frage beantworten Allgemeine Java-Themen 1
pkm Frage zur Präzision von Calendar.WEEK_OF_YEAR Allgemeine Java-Themen 12
J Eine Frage zu den Threads und Task Allgemeine Java-Themen 1
pkm Frage nach eventuellem syntaktischen Zucker bei der Konkatenation von ArrayLists Allgemeine Java-Themen 4
M Frage-Antwortspiel wie Wer wird Millionär Allgemeine Java-Themen 1
F Frage zu System.in Allgemeine Java-Themen 3
marcooooo Frage zum Beispiel im Anhang Allgemeine Java-Themen 16
T Meine Frage lautet wie ich 2 CSV Dateien miteinander in Java verbinde und Spalten die zueinander gehören durch den gleichen Key zusammen ausgebe? Allgemeine Java-Themen 5
S Noch eine Design-Frage zu Setter Allgemeine Java-Themen 6
B For-Loop Frage Allgemeine Java-Themen 21
L Java frage Allgemeine Java-Themen 3
bueseb84 Frage zu Mock und UpperBound Allgemeine Java-Themen 2
M Frage zum Konstruktor Allgemeine Java-Themen 2
W Best Practice Frage zur Umsetzung MVC Allgemeine Java-Themen 9
P String-Verschlüsselung - Frage zur Sicherheit Allgemeine Java-Themen 21
B Frage zu Unit-Tests Allgemeine Java-Themen 6
T Allgemeine Frage: GUI für 3D-Visualisierung Allgemeine Java-Themen 5
R Allgemeine Frage zu RMI bei MVC Allgemeine Java-Themen 2
O Frage zum Runtimeverhalten von Java ... Allgemeine Java-Themen 2
H Rundreise frage (Algorithmus) Allgemeine Java-Themen 18
B Generelle Frage bei einer Webanwendung / Reduzierung von DB Abfragen Allgemeine Java-Themen 1
D Frage zu Vererbung Allgemeine Java-Themen 5
J Frage zu regulärem Ausdruck Allgemeine Java-Themen 2
M Allgemeine Frage: Wie lernt man Java / Programmieren von Grund auf? Allgemeine Java-Themen 7
S Frage zur JLS Allgemeine Java-Themen 0
J Verständnis Frage zur Instanz, Objekte, Instanzierung, Referenz Allgemeine Java-Themen 14
A Methoden Allgemeine Java Frage Allgemeine Java-Themen 3
E String Frage Allgemeine Java-Themen 9
I bin neu bei GitHub, Frage zur Sicherheit Allgemeine Java-Themen 14
C J2V8 NodeJs Java Bride Problem und Frage!?!? Allgemeine Java-Themen 1
C KeyListener Frage Allgemeine Java-Themen 3
T Frage zu UML in Java programmieren Allgemeine Java-Themen 1
R Konstanten initialisieren - FRAGE Allgemeine Java-Themen 3
MTJ004 FTP Frage zu FTP Speicherung Java-Android-FTP Allgemeine Java-Themen 5
J Frage zum Entwurf / json-Datenmodell Allgemeine Java-Themen 8
A Frage zu meinem Code Allgemeine Java-Themen 2
RalleYTN Classpath Nur ne kleine Frage zur MANIFEST.MF Allgemeine Java-Themen 4
T Frage zu Access Modifiers Allgemeine Java-Themen 6
W Input/Output Frage zu pdfbox und FileUtils Allgemeine Java-Themen 2
O Frage zur Implementierungsweise Allgemeine Java-Themen 4
B Frage zu Bitshift Allgemeine Java-Themen 3
J Java Zufallsgenerator (6 aus 49) Frage Allgemeine Java-Themen 7
L Frage zu RIA und GWT Allgemeine Java-Themen 0
P Concurrency Frage Allgemeine Java-Themen 8
M Frage zu Enumerations Allgemeine Java-Themen 2
F Unlimited Strength Policy. Frage Verbreitung der Anwendung Allgemeine Java-Themen 1
F Frage zur Library JTS Allgemeine Java-Themen 5
S Java Design Frage Allgemeine Java-Themen 10
E Reflection? Frage Allgemeine Java-Themen 4
C FileInputStream frage Allgemeine Java-Themen 6
G Polymorphie Programmdesign Frage Allgemeine Java-Themen 20
Uzi21 Frage zu NetBeans ( Console) Allgemeine Java-Themen 11
D Classpath Frage zum Java Resource Loading Allgemeine Java-Themen 2
G Frage zu JPA Allgemeine Java-Themen 1
S Methoden Frage Allgemeine Java-Themen 2
P MVC - Frage zu Model Allgemeine Java-Themen 4
K Frage zu Locks Allgemeine Java-Themen 1
S Frage zu abstract Allgemeine Java-Themen 5
M ArrayList<String> Frage Allgemeine Java-Themen 7
M OOP Design Frage Allgemeine Java-Themen 2
N Frage zur while-Schleife Allgemeine Java-Themen 18
T Best Practice Auslesen von Zeichenketten (Frage, Antworten, usw) Allgemeine Java-Themen 4
C Eine Frage zur Bearbeitungszeit Allgemeine Java-Themen 8
H Frage wegen Heap-Speicher Allgemeine Java-Themen 2
T Garbage Collection Frage Allgemeine Java-Themen 15
P Kurze Frage: aus einer File die Zeilenanzahl auslesen Allgemeine Java-Themen 9
D Frage zu Java und Umlauten / charsets Allgemeine Java-Themen 2
B Frage zu Java und OpenGL? Allgemeine Java-Themen 3
Q Kapselung Allgemeine Design- Frage Allgemeine Java-Themen 8
A eine test thread.join() frage Allgemeine Java-Themen 2
DStrohma LayoutManager Frage zum GridBagLayout Allgemeine Java-Themen 4
F Frage zu Regex möglich Allgemeine Java-Themen 4
H XML-File mit Java erzeugt Frage Allgemeine Java-Themen 10
D Frage und Antwort Programm, Problem bei Methodenaufruf Allgemeine Java-Themen 3
J NetBeans Frage bezüglich der Scanner-Klasse Allgemeine Java-Themen 6
H Java Vector Frage Allgemeine Java-Themen 9
W Frage... Allgemeine Java-Themen 29
R Frage zur topologischen Sortierung Allgemeine Java-Themen 2
H Frage zu weka.core.Instance Allgemeine Java-Themen 3
Y Kleine Frage zu String.split Allgemeine Java-Themen 3
T Frage zu Klassendesing Allgemeine Java-Themen 3
W Frage zu Refactoring statischer Methoden Allgemeine Java-Themen 4
C Eclipse Wichtige frage Allgemeine Java-Themen 5
H Frage zu java.weka.core.Instances Allgemeine Java-Themen 3
S Frage zu Format Modifiers in Log4j Allgemeine Java-Themen 11
H Frage zu clone() Allgemeine Java-Themen 5
4 Simple(?) Frage zu Threads Allgemeine Java-Themen 14
H2SO3- SCJP Chapter 3 Frage 10. Falsche Antwort? Allgemeine Java-Themen 15

Ähnliche Java Themen

Neue Themen


Oben