Spezifischte Methode

Leyla

Mitglied
Kann mir jemand erklären, warum es hier einen Compiler-Fehler gibt? Ich versuche gerade herauszufinden, wie Java die spezifischste Methode findet.
WhatsApp Image 2023-11-02 at 15.00.38.jpeg

Warum werden hier zwei Methoden gecalled. Also woran liegt das?
 

httpdigest

Top Contributor
Die genauen Regeln stehen in der Java Language Specification: https://docs.oracle.com/javase/specs/jls/se8/html/jls-15.html#jls-15.12.2.5
Insbesondere der Punkt:
  • m2 is not generic, and m1 and m2 are applicable by strict or loose invocation, and where m1 has formal parameter types S1, ..., Sn and m2 has formal parameter types T1, ..., Tn, the type Si is more specific than Ti for argument ei for all i (1 ≤ in, n = k).
ist hier nicht erfüllt, weil die eine der beiden anwendbaren "max" Methoden laut dieser Definition nicht spezifischer ist als die andere der beiden anwendbaren "max" Methoden.
Damit eine nicht-generische Methode m1 spezifischer ist, als eine andere m2, muss die oben genannte Bedingung für alle formalen Parameter gelten.
Da hängt es dann nicht davon ab, ob nun mehr oder weniger Casts für die Argumente des Aufrufes gemacht werden müssen, sondern nur, ob es eine eindeutige Subtyp-Beziehung zwischen den formalen Parametern der anwendbaren Methoden gibt.

Die Typen der aktuellen Argumente spielen nur eine Rolle beim Herausfinden, welche Methoden prinzipiell anwendbar sind. Nicht aber mehr beim Ermitteln, was nun die spezifischste Methode ist.
 

KonradN

Super-Moderator
Mitarbeiter
Es werden nicht zwei Methoden aufgerufen sondern zwei Methoden passen gleich gut.

Du hast als Parameter: short, int, int
Eine solche Methode gibt es nicht. Also schaut er, ob er durch implizite casts eine Methode aufrufen könnte:

Bei int, long, int würde es gehen, denn ein short kann implizit in ein int und ein int kann implizit in ein long gecastet werden.
Bei short, int, double geht es auch, denn ein int kann implizit in ein double gecastet werden.
Somit gibt es zwei Methoden, die hier in Frage kommen können und daher kann es der Compiler nicht für Dich entscheiden.

Das wäre die ganz vereinfachte Darstellung. Die genauen Details finden sich unter:
 

Leyla

Mitglied
Danke für die schnelle Antwort.
@KonradN
Aber dann würde doch eigentlich auch die erste Methode passen. Weil ein "short" zu "long" gecastet werden kann und die beiden "ints" auch. Oder?
@httpdigest
Könntest du diese Regel mit Beispielen erklären? Ich bin neu im Programmieren und mein Prof hat das heute erklärt, aber hab nichts verstanden und auch durch die ganzen Youtube-Beiträge. Niemand erklärt das.
 

KonradN

Super-Moderator
Mitarbeiter
Aber dann würde doch eigentlich auch die erste Methode passen. Weil ein "short" zu "long" gecastet werden kann und die beiden "ints" auch. Oder?
Ja, die würde auch passen. Kommentiere eine der beiden genannten Methoden um und du wirst die verbliebenen zwei Methoden bekommen.

Das ist dann eine Sache, die nichts mit der Spezifikation sondern aus meiner Sicht mit dem Compiler zu tun hat. Wenn er zwei Methoden gefunden hat, dann reicht es ihm und er gibt die Fehlermeldung aus.

Bei IntelliJ zeigt er da dann auch an, dass es eben diese 3 Kandidaten gibt.

Du kannst damit ja weiter herum spielen und noch mehr passende Methoden erzeugen. short, double, double und so. Er wird immer nur zwei Methoden anzeigen bei der Fehlermeldung.

Edit: Evtl. etwas genauer zu "wenn er zwei Methoden gefunden hat": Er hat keine genau passende Methode und zwei Methoden, die mit impliziten Casts passen würden. Die genau passende Methode löst den Fehler sofort: erstelle eine max(short, int, int) Methode und alles ist gut.
 

Robert Zenz

Top Contributor
Danke für die schnelle Antwort.
@KonradN
Aber dann würde doch eigentlich auch die erste Methode passen. Weil ein "short" zu "long" gecastet werden kann und die beiden "ints" auch. Oder?
@httpdigest
Könntest du diese Regel mit Beispielen erklären? Ich bin neu im Programmieren und mein Prof hat das heute erklärt, aber hab nichts verstanden und auch durch die ganzen Youtube-Beiträge. Niemand erklärt das.
Also, du hast die Signatur short/int/int, fuer long/long/long sind das 3 Umwandlungen, fuer int/long/int sind es 2 Umwandlungen und fuer short/int/double sind es auch zwei Umwandlungen. Damit werden dir die letzten beiden Methoden angezeigt/vorgeschlagen weil diese am besten passen, und die long/long/long Methode passt nicht so gut wie die anderen. Du koenntest auch die long/long/long Methode mit deinen Parametern aufrufen, aber die anderen beiden Methoden brauchen weniger Umwandlungen und damit geht die IDE davon aus dass du eine von den beiden wolltest.
 

Leyla

Mitglied
Ok, das leuchtet soweit ein. Aber wieso kommt dann hier jetzt dann eine Fehlermeldung?
Java:
public class test {
    public static void max(long a, long b, long c) {

    }
    public static void max(int a, long b, int c) {
    }
    public static void max(short a, int b, double c) {

    }
    public static void main(String[] args) {
        max((byte) 1, (byte) 1,1L);

    }

}

Methode 1: zwei Umwandlungen
Methode 2: drei Umwandlungen
Methode 3: drei Umwandlungen

Aber es kommt folgende Fehlermeldung:
Ambiguous method call. Both
max(long,long,long) in test and max(short,int,double) in test match

Aber eigentlich sollte ja in dem Fall Methode 1 aufgerufen werden oder nicht ?

Nebenbei angemerkt, ich danke jedem hier, der mir gerade hilft! <3
 

Leyla

Mitglied
Moment, verstehe ich Sie richtig. Findet kein Casting statt, sondern nur eine Erweiterung. Somit gilt für die Methode 3 nur zwei Umwandlungen, weil Long zu double erweitert werden kann ?

Bei welchen Datentypen wäre das auch so ?
 

mihe7

Top Contributor
Erweiterung und Casting sind beides Umwandlungen. Casting bedeutet nur, dass die Umwandlung explizit unter Zuhilfenahme des Casting-Operators angegeben wird.

In der Zeile
Code:
max( (byte) 1, (byte) 1, 1L)
hast Du zwei int-Literale und ein long-Literal stehen. Die beiden int-Literale castest Du auf byte. Du könntest den Code ersetzen durch:
Code:
byte a = (byte) 1;
byte b = (byte) 1;
long c = 1L;
max(a, b, c);
Der Compiler sieht jetzt: aha, zwei byte und ein long, welche Methoden stehen mir zur Verfügung, wenn nach 5.1.2 folgende Umwandlungen implizit erlaubt sind?

byte -> byte, short, int, long, float, or double
long -> long, float, double

Bewertung:
Code:
max(long a, long b, long c) -> geht, weil byte zu long implizit erweitert werden kann
max(int a, long b, int c) -> geht nicht, weil long nicht zu int implizit erweitert werden kann
max(short a, int b, double c) -> geht, weil byte zu short, byte zu int und long zu double implizit erweitert werden kann
 

Leyla

Mitglied
Erweiterung und Casting sind beides Umwandlungen. Casting bedeutet nur, dass die Umwandlung explizit unter Zuhilfenahme des Casting-Operators angegeben wird.

In der Zeile
Code:
max( (byte) 1, (byte) 1, 1L)
hast Du zwei int-Literale und ein long-Literal stehen. Die beiden int-Literale castest Du auf byte. Du könntest den Code ersetzen durch:
Code:
byte a = (byte) 1;
byte b = (byte) 1;
long c = 1L;
max(a, b, c);
Der Compiler sieht jetzt: aha, zwei byte und ein long, welche Methoden stehen mir zur Verfügung, wenn nach 5.1.2 folgende Umwandlungen implizit erlaubt sind?

byte -> byte, short, int, long, float, or double
long -> long, float, double

Bewertung:
Code:
max(long a, long b, long c) -> geht, weil byte zu long implizit erweitert werden kann
max(int a, long b, int c) -> geht nicht, weil long nicht zu int implizit erweitert werden kann
max(short a, int b, double c) -> geht, weil byte zu short, byte zu int und long zu double implizit erweitert werden kann


Code:
max(long a, long b, long c) -> geht, weil byte zu long implizit erweitert werden kann
max(int a, long b, int c) -> geht nicht, weil long nicht zu int implizit erweitert werden kann.
max(short a, int b, double c) -> geht, weil byte zu short, byte zu int und long zu double implizit erweitert werden kann

Bei max(int a, long b, int c) wird doch aber nicht long nach int impliziert erweitert, sondern umgekehrt. Oder nicht ? Der erste Parameter wird von byte zu int impliziert erweitert. Der zweite Parameter wird von byte nach long erweitert und der letzte Parameter wird von int nach long erweitert oder?
 

Leyla

Mitglied
Was ich mich dann aber frage ist, warum es zum Beispiel zu keinem Compilerfehler kommt, wenn man die Methode mit max(1,1,1) aufruft, weil hier hätte man auch drei Kandidaten.
 

Robert Zenz

Top Contributor
Was ich mich dann aber frage ist, warum es zum Beispiel zu keinem Compilerfehler kommt, wenn man die Methode mit max(1,1,1) aufruft, weil hier hätte man auch drei Kandidaten.
Ja, aber nur eine davon braucht eine Erweiterung/Umwandlung, naemlich int/long/int, damit gewinnt diese Methode die Bewertung welcher am besten passt. Die mit short faellt komplett raus (wein implizit von int auf short geht nicht) und long/long/long braucht drei Erweiterungen/Umwandlungen.

Du bekommst ja nur dann einen Fehler wenn es mehrere Methoden gibt welche passen wuerden und die gleiche Anzahl an UMwandlungen haben damit sie passen.
 

mihe7

Top Contributor
Bei max(int a, long b, int c) wird doch aber nicht long nach int impliziert erweitert, sondern umgekehrt. Oder nicht ? Der erste Parameter wird von byte zu int impliziert erweitert. Der zweite Parameter wird von byte nach long erweitert und der letzte Parameter wird von int nach long erweitert oder?
Zur ersten Frage: ganz genau. int ist keine Erweiterung von long, daher ist die Umwandlung nicht implizit möglich.

Zweite Frage: nein, der letzte Parameter müsste von long (es ist ja ein long-Literal gegeben) nach int (Typ des Parameters der Methode) umgewandelt werden. Eine solche "Verengung" des Datentyps ist implizit nicht möglich, daher scheidet diese Methode aus.
 

LimDul

Top Contributor
0=pass directly
1=implicit cast
-1=doesn't pass / explicit cast needed

Sobald einmal -1 vorkommt, wird diese Methode als möglicher Aufruf ausgeschlossen. Wenn es aber mehrere Methoden gibt, für die alle Argumente >= 0 gelten, wird nicht mehr zwischen "direct pass" und "implicit cast" unterschieden, sondern sie wird als "ambiguous" behandelt. Mehrere mögliche Methoden ("ambiguous") darf es für einen Aufruf nicht geben und ist in der Folge ein Compilierfehler zur Übersetzungszeit.

Ich denke, das erklärt die Frage.
Die Tabelle ist gut, aber deine Aussage ist meines Erachtens falsch.

Wenn es eine Methode gibt, wo nur pass directly gilt, wird diese genommen.
Dann interessiert der Rest nicht mehr. Nur wenn es diese nicht gibt, wird geschaut, ob es mehrere Kandidaten gibt, die einen impliziten Cast brauchen. Wenn es da zwei gibt => Error.
 
Ähnliche Java Themen
  Titel Forum Antworten Datum
T Rekursive Methode Java Basics - Anfänger-Themen 13
Ü Methode soll Quadrat aus der Summer zurückgeben Java Basics - Anfänger-Themen 10
P Objekt einer Methode eines anderen Objektes übergeben Java Basics - Anfänger-Themen 5
M Methode zielnah zeigt das gewünschte Ausgabe nicht an Java Basics - Anfänger-Themen 3
L Variablenwerte aus einer Methode übergeben Java Basics - Anfänger-Themen 2
T Methode soll etwas ausrechnen und zurückgeben (klappt nd) hat wer eine Idee? Java Basics - Anfänger-Themen 11
P Main Methode scheint Constructor aufzurufen, ohne dass es so gecoded ist Java Basics - Anfänger-Themen 2
T Aufruf der Methode einer Oberklasse, wenn sie in der Unterklasse überschrieben ist. Polymorphie. Java Basics - Anfänger-Themen 2
C Zugriff auf Methode Java Basics - Anfänger-Themen 2
M Datentypen While-Schleife eine Java Methode erstellen Java Basics - Anfänger-Themen 3
T Methode akzeptiert String nicht Java Basics - Anfänger-Themen 18
M Methode sperren bis ein Kriterium erfüllt wurde Java Basics - Anfänger-Themen 3
D Switch Case Methode aufrufen Java Basics - Anfänger-Themen 3
C Unbekannte Methode add bei Klasse die JTree erweitert Java Basics - Anfänger-Themen 14
M methode aufrufen ohne parameter Java Basics - Anfänger-Themen 1
marcelnedza Finde meinen Fehler in einer Methode nicht, Java Karol Java Basics - Anfänger-Themen 15
monsterherz einfache Methode mit Fehler den ich nicht finde Java Basics - Anfänger-Themen 21
Ostkreuz Wieso wird die Methode nochmal aufgerufen? Java Basics - Anfänger-Themen 5
G Variable aktualisiert sich nicht in rekursiver Methode Java Basics - Anfänger-Themen 4
MoxMorris Wie macht man String[] = String[] aus einer anderer Methode? Java Basics - Anfänger-Themen 18
Say super.methode / super.variable und super(variable) Java Basics - Anfänger-Themen 2
B Wie kann ich folgende Klasse/Methode per Button ausführen? Java Basics - Anfänger-Themen 1
D Interface Methode wird ungewollt in der Subklasse überschrieben Java Basics - Anfänger-Themen 5
L Methoden Eine Methode um zu testen ob es ein Nachbar gibt Java Basics - Anfänger-Themen 10
til237 Iterative Methode in rekursive Methode umschreiben Java Basics - Anfänger-Themen 4
M Daten aus errechneter Methode in Datenbank(SQLite) schreiben Java Basics - Anfänger-Themen 60
D next() Methode mehrfach verwenden Java Basics - Anfänger-Themen 1
Ostkreuz Methoden Von Dezimal zu Hexadezimal Methode toHex Java Basics - Anfänger-Themen 2
I Entity Objekt nicht gefunden -> Webhook empfangen in der gleichen Methode (Transaktion) Java Basics - Anfänger-Themen 37
N Throw an Main Methode übergeben Java Basics - Anfänger-Themen 7
M Methoden Methode 'wiederhole' nicht gefunden (Uebersetzungsfehler) Java Basics - Anfänger-Themen 1
H Zu langen String aufteilen - bequeme Methode? Java Basics - Anfänger-Themen 14
_user_q Wie eine Methode/Funktion aus einer Klasse mit Constructor aufrufen? Java Basics - Anfänger-Themen 20
S Array mit Methode löschen Java Basics - Anfänger-Themen 2
J Java To String Methode, Array mit For-Schleife Java Basics - Anfänger-Themen 2
T Variable von Objekten in einer Methode überprüfen Java Basics - Anfänger-Themen 26
M Anzahl Kommandozeilenparamter mittels Methode Java Basics - Anfänger-Themen 11
D Methode: Array Reihenfolge tauschen Java Basics - Anfänger-Themen 3
julian0507 Array aus Methode in anderer Methode sichtbar machen Java Basics - Anfänger-Themen 10
frager2345 Problem mit Methode Java Basics - Anfänger-Themen 4
J Die statische Main-Methode ändert Instanzvariable? Java Basics - Anfänger-Themen 10
D Methode aus dem Aufrufer aufrufen Java Basics - Anfänger-Themen 1
T IOStreams read(byte[]b) methode Java Basics - Anfänger-Themen 2
frager2345 Java Singleton Muster -> Methode für Konstruktor mit Parametern Java Basics - Anfänger-Themen 3
U Beispiel Methode size() vom "Collection"-interface... Wie kann man sichtbar machen, was die Methode unter der Haube macht? Java Basics - Anfänger-Themen 8
D Warum kann ich hier nicht auf die Methode zugreifen? Java Basics - Anfänger-Themen 5
M generate Methode für Streams Java Basics - Anfänger-Themen 6
M Methoden Zweidimensionaler Array mit Setter Methode ändern Java Basics - Anfänger-Themen 4
I Optionaler Parameter bei Methode, der nur optional ist? Java Basics - Anfänger-Themen 6
berserkerdq2 Wozu benötigt man den BiPredicate, kann ich nicht einfach eine normale Methode nutzen, statt BiPredicate? Java Basics - Anfänger-Themen 3
T Linked List set-Methode Java Basics - Anfänger-Themen 2
D Arrays an replaceAll-Methode übergeben Java Basics - Anfänger-Themen 12
B Attribute eines Objekts einer Klasse durch statische Methode einer 2. Klasse ändern? Java Basics - Anfänger-Themen 32
berserkerdq2 Habe eine Klasse, welche public ist, diese hat eine public Methode, die nicht static ist. Wenn ich nun versuche aufzurufen Probleme? Java Basics - Anfänger-Themen 8
viktor1 Methoden Methode schreiben static void readText (String filename) {...} zu WordHistogramSample.java Java Basics - Anfänger-Themen 13
W Equals-Methode überschreiben bei composition Java Basics - Anfänger-Themen 20
V Hilfe bei Implementierung einer boolean Methode Java Basics - Anfänger-Themen 6
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
V Methoden printChar Methode mit Rückgabetyp void Java Basics - Anfänger-Themen 26
F Graph Tiefensuche Methode Java Basics - Anfänger-Themen 7
Jambolo Methode, welche die 3 letzten Parameter Werte speichert Java Basics - Anfänger-Themen 20
berserkerdq2 wie funktioniert contenthandler, was muss ich bei der Methode startElement und endElement tun? Java Basics - Anfänger-Themen 11
M Warum return die Methode den Wert nicht Java Basics - Anfänger-Themen 5
berserkerdq2 Wann soll ich den Stream schließen, wenn ich das in einer Methode habe? Java Basics - Anfänger-Themen 8
berserkerdq2 Ich gebe eine ArrayList als List zurück per MEthode, wie kann ich nun aber die ArrayList speichern? Java Basics - Anfänger-Themen 46
S Methode Java Basics - Anfänger-Themen 4
M Eine Methode die erkennt ob die ein gegebene zahl größer oder kleiner sein muss Java Basics - Anfänger-Themen 2
U Methode wird genutzt, ohne dass ich die aufrufe? Java Basics - Anfänger-Themen 4
F nach Methode Programm nicht beenden Java Basics - Anfänger-Themen 9
Liroyd Methode mit Objektvariabel rechnen? Java Basics - Anfänger-Themen 6
H Mit setter-Methode JLabel in einer andern Klasse ändern. Java Basics - Anfänger-Themen 40
D Methode um mögliche Rezepte auszugeben Java Basics - Anfänger-Themen 12
U Warum kann ich die Methode in der ENUM Klasse nicht aufrufen? Und warum geht die Switch nicht? Java Basics - Anfänger-Themen 8
J Hallo zusammen , was macht diese Methode hier genau? Java Basics - Anfänger-Themen 3
D Array in Main Methode aus anderer Klasse aufrufen Java Basics - Anfänger-Themen 3
H Eine Methode über Actionlistener beenden Java Basics - Anfänger-Themen 8
G jButton führt Main Methode nicht richtig aus Java Basics - Anfänger-Themen 3
G Main Methode wird beim ersten Aufruf nicht richtig ausgeführt Java Basics - Anfänger-Themen 1
C60 Methoden Main-Methode erkennt meine Arrays nicht. Java Basics - Anfänger-Themen 7
A Ein Array bearbeiten und in einer anderen Methode nutzen Java Basics - Anfänger-Themen 6
A Ergebnis einer Methode bei einer anderen verwenden Java Basics - Anfänger-Themen 13
L Iteratorform und Methode mit variabler Parameterzahl Java Basics - Anfänger-Themen 31
F Methode ArrayList mit Eingabewert Java Basics - Anfänger-Themen 2
M Wie kann eine Methode für ein vorhandenes "Array von char" einen Index-Wert zurückliefern? Java Basics - Anfänger-Themen 3
M Wie kann die Implementation einer Methode den Wert eines Attributs vermindern? Java Basics - Anfänger-Themen 3
Csircc Rekursive Methode Stack Overflow Java Basics - Anfänger-Themen 10
M Wie kann eine Methode (string) eine andere Methode (void) mit zufälligen int-Werten aufrufen? Java Basics - Anfänger-Themen 4
M Wie verknüpfe ich eine Bedingung mit einer Methode ohne if-Verzweigung & Bedingungsoperator? Java Basics - Anfänger-Themen 2
M Wie kann eine Methode eine andere Methode um Werte wie z.B. 1 erhöhen? Java Basics - Anfänger-Themen 6
schredder Strings und reguläre Ausdrücke - Methode mit return string.matches Java Basics - Anfänger-Themen 5
D mehrere Berechnungen in einer Methode Java Basics - Anfänger-Themen 9
H String Repräsentation eines Rechtecks mit Instanz-Methode Java Basics - Anfänger-Themen 8
M Wie kann ich eine Methode aus einem Interface in eine Klasse implementieren, so dass sie ihre Funktion ausführt? Java Basics - Anfänger-Themen 7
J ArrayList add methode selbst programmieren Java Basics - Anfänger-Themen 10
X Methoden Methode zur Punktezählung in Blackjack Java Basics - Anfänger-Themen 2
W Methode ändern Java Basics - Anfänger-Themen 65
M Wie kann ich in einem Konstruktor die Methode eines anderen Interfaces mit den jeweiligen Parametern aufrufen? Java Basics - Anfänger-Themen 8
W Methode, die mit einem Datum arbeitet? Java Basics - Anfänger-Themen 22
E Einrücken inner der toString()-Methode Java Basics - Anfänger-Themen 34
C Java Arrays - Ausgabe in Methode Java Basics - Anfänger-Themen 12

Ähnliche Java Themen

Neue Themen


Oben