array.sort mit zwei Kriterien

Thongall

Mitglied
Hallo,
ich habe ein Problem beim Sortieren von Objekt Arrays.

Ich möchte ein Array von Objekten mit zwei comparatoren sortieren.

Dazu vergleiche ich in jedem comparator den Wert nach dem Sortiert werden soll.
Im Hauptprogramm rufe ich dann Array.sort mit dem jeweiligen comparator auf.
Soweit kein Problem. Allerdings möchte ich das wenn ich nach dem ersten Wert sortiere und der Wert bei 2 Objekten gleich ist dann als zweites Kriterium den zweiten comparator nutzen.

Meine Idee war nun das sich die comparatoren einfach jeweils gegenseitig aufrufen.
Das klappt auch soweit außer natürlich wenn die Werte bei beiden comparatoren gleich sind - dann endet die Rekursion nie. Mein Problem ist nun das ich nicht weiß wo ich die return 0 zum Abbrechen einfügen soll oder ob es dafür generell eine andere Möglichkeit gibt.
Denn setz ich das return 0 vor dem rekursiven Aufruf wird dieser niemals ausgeführt, setz ich sie danach wird das return 0 niemals erreicht bei zwei gleichen werten.

über einen kleinen Denkanstoß wäre ich sehr dankbar.

MfG
Thongall
 

Joose

Top Contributor
Deine Idee klingt schon richtig. Der Fehler ist nur
.... das sich die comparatoren einfach jeweils gegenseitig aufrufen ....
das klingt schon nach Endlosschleife ;)

Du vergleichst den 1.Wert von 2 Objekten, wenn dieser gleich ist wird der andere Comparator aufgerufen. Dieser vergleicht nun den 2.Wert, wenn dieser wieder gleich ist dann musst du entscheiden welches nun "größer"/"kleiner" ist. Also entweder lässt du die Reihenfolge so wie sie ist wenn beide Werte gleich sind, oder du tauschst sie einfach.
 
Zuletzt bearbeitet:

Flown

Administrator
Mitarbeiter
Ohne Code kann dir hier keiner helfen!

Du hast Glück das ich so ähnlichen Code herumliegen habe und ich dir ein kleines KSKB basteln konnte:

Java:
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;
import java.util.Random;

public class TupleCompareTest {
	public static void main(String[] args) {
		new TupleCompareTest();
	}
	
	public TupleCompareTest() {
		List<Tuple> tuples = new ArrayList<>();
		for(int i = 0; i < 10; i++) {
			tuples.add(new Tuple(createRandomInt(), createRandomInt()));
		}
		printList(tuples);
		Collections.sort(tuples, new TupleComparatorX(new TupleComparatorY()));
		printList(tuples);
	}
	
	private int createRandomInt() {
		Random rnd = new Random();
		int nr = rnd.nextInt(10);
		return rnd.nextBoolean() ? nr : Math.negateExact(nr);
	}
	
	private void printList(List<Tuple> list) {
		list.forEach(e -> System.out.print(e + " "));
		System.out.println();
	}
	
	class Tuple {
		public int x, y;
		public Tuple(int x, int y) {
			this.x = x;
			this.y = y;
		}
		@Override
		public String toString() {
			return String.format("(%d, %d)", x, y);
		}
	}
	
	class TupleComparatorY implements Comparator<Tuple>{
		@Override
		public int compare(Tuple o1, Tuple o2) {
			return o1.y - o2.y;
		}
	}
	
	class TupleComparatorX implements Comparator<Tuple>{
		
		private TupleComparatorY tcy;
		
		public TupleComparatorX(TupleComparatorY tcy) {
			this.tcy = tcy;
		}
		
		@Override
		public int compare(Tuple o1, Tuple o2) {
			if(o1.x == o2.x) return tcy.compare(o1, o2);
			return o1.x - o2.x;
		}
	}
}
 
Zuletzt bearbeitet:

Thongall

Mitglied
Vielen Dank schonmal für die Antworten - versteh jetzt schon deutlich mehr wie eben über die Funktionsweise. Ein Problem besteht aber weiterhin.

Ich würde gerne entweder nach comperator a oder nach comperator b soriteren lassen und dann zusätlich jeweils nach dem anderem als zweites Kriterium.

Bei dem Beispiel bzw. der Erklärung klappt das immer nur in eine Richtung - hab es versucht ein wenig zu modifizieren aber lande dann wieder in der Endlos Rekursion... :/

Als Code Beispiel hab ich jetzt den Code von Flown. Dort kann man nach x und dann y sortieren oder wenn man es umschreibt umgekehrt - aber nicht beides gleichzeitig.

Also z.b. Hab ich ein Array und mache davon eine Kopie. Das Original will ich nach x und dann nach y Sortieren lassen und die Kopie nach y und dann nach x.

Edit:
Habs gelöst :)
Man muss gar nicht den anderen Comperator aufrufen sondern einfach beide vergleiche in der Reihenfolge wie man sie haben will im jeweiligen Comperator durchführen.
Thema ist damit erledigt, vielen Dank für die Hinweise :)
 
Zuletzt bearbeitet:

Joose

Top Contributor
Ich würde gerne entweder nach comperator a oder nach comperator b soriteren lassen und dann zusätlich jeweils nach dem anderem als zweites Kriterium.

Bei dem Beispiel bzw. der Erklärung klappt das immer nur in eine Richtung - hab es versucht ein wenig zu modifizieren aber lande dann wieder in der Endlos Rekursion... :/

......................
Habs gelöst :)
Man muss gar nicht den anderen Comperator aufrufen sondern einfach beide vergleiche in der Reihenfolge wie man sie haben will im jeweiligen Comperator durchführen.
Thema ist damit erledigt, vielen Dank für die Hinweise :)

(wenn ich die richtig verstanden habe ;-))

Nein so kombinierst du die Vergleiche nicht wirklich schon miteinander! Für jede Kombination müsstest du dann einen eigenen Comparator schreiben, was vielleicht unübersichtlich und sehr viel werden könnte.

Schöner wäre eine abstrakte Basisklasse für deine Comparatoren, diese implementiert das Interface Compareable und kann ein Objekt von sich selber speichern.
In den spezifischen Klassen, implementierst du die compare Methode, und wenn für 2 Objekte der geprüfte Wert gleich ist, dann prüfst du ob der Comparator einen Referenz auf einen weiteren hat, wenn ja leitest du den vergleich weiter, ansonsten gibst du 0 zurück(oder was eben rauskommt).

So kannst du die Reihenfolge in der die Werte überprüft werden einfach ändern ohne die Methoden anzupassen
 

Thongall

Mitglied
Gut das wäre natürlich eine noch bessere Lösung - hab auch gar nicht an mehr als 2 vergleiche gedacht aber stimmt das wird dann irgendwann sehr unschön... danke für den Hinweis :)
 

strußi

Top Contributor
sortier doch erst nach dem ersten kriterium und
lauf dann mit einem for-schleife über die objects drüber und schau wenn obj1.kriterium =obj2.kriterium
k2o1 =getobj1.kriterium 2 und k2o2 =getobj2.kriterium k2o1 <= |> k2o2
 

DrZoidberg

Top Contributor
Hier ist noch ein anderer Ansatz mit lambdas.

Damit kannst du eine Liste nach beliebig vielen Kriterien sortieren.

Java:
import java.util.Arrays;
import java.util.ArrayList;
import java.util.List;
import java.util.Collections;
import java.util.Comparator;

public class Test {
    public static void main(String[] args) {
        List<Person> list = Arrays.asList(
            new Person("peter", "griffin"),
            new Person("homer", "simpson"),
            new Person("kyle", "broflovski"));
        
        Collections.sort(list, Comparator.comparing(p -> new CompList(p.lastName, p.firstName)));
        
        for(Person p: list) System.out.println(p);
    }
    
    static class Person {
        String firstName;
        String lastName;
        Person(String firstName, String lastName) {
            this.firstName = firstName;
            this.lastName = lastName;
        }
        
        @Override
        public String toString() {
            return firstName + " " + lastName;
        }
    }
    
    static class CompList implements Comparable<CompList> {
        ArrayList<Comparable<?>> list = new ArrayList<>();
        CompList(Comparable<?> ... elems) {
            for(Comparable<?> elem: elems) list.add(elem);
        }
        
        @SuppressWarnings("unchecked")
        public int compareTo(CompList other) {
            for(int i = 0; i < list.size() && i < other.list.size(); i++) {
                Comparable elem1 = list.get(i);
                Comparable elem2 = other.list.get(i);
                int c = elem1.compareTo(elem2);
                if(c < 0) return -1;
                else if(c > 0) return 1;
            }
            if(list.size() < other.list.size()) return -1;
            else if(list.size() > other.list.size()) return 1;
            else return 0;
        }
    }
}
 

nvidia

Bekanntes Mitglied
Würde ich eine Liste von Personen haben und die nach Vor-/Nachname aufsteigend sortieren wollen, d.h. aus [(c,b) (c,a)] soll die Reihenfolge [(c,a) (c,b)] werden, wobei der erste Wert des Tupels dem Nachnamen und der zweite Wert des Tupels dem Vornamen entspricht, würde ich folgende Zeile verwenden.

Java:
 list.sort(comparing(Person::lastName).thenComparing(Person::firstName));
 
Zuletzt bearbeitet:
Ähnliche Java Themen
  Titel Forum Antworten Datum
M Bubble Sort - Int[] Array sortieren Java Basics - Anfänger-Themen 2
L Insertion Sort bei zweidimensionalem Array Java Basics - Anfänger-Themen 7
KogoroMori21 Textdatei einlesen im Array (Selection Sort Algorithmus) Java Basics - Anfänger-Themen 3
S Methoden Sort Array Java Basics - Anfänger-Themen 9
U Methoden Zweidimensionales Array mit Arrays.sort sortieren? Java Basics - Anfänger-Themen 22
C OOP array Sortieren ohne den sort Befehl Java Basics - Anfänger-Themen 10
S int-Array mittels Arrays.sort() in einer Schleife sortieren. Java Basics - Anfänger-Themen 2
S Fehler bei Arrays.sort(array) - Methode!? Java Basics - Anfänger-Themen 3
K Array.sort() Java Basics - Anfänger-Themen 12
B 2 dimensionales Array: Selection Sort Java Basics - Anfänger-Themen 4
P Array.sort // Arrays ausgeben Java Basics - Anfänger-Themen 21
I Selection-Sort // Array *help* Java Basics - Anfänger-Themen 2
H Bubble sort array Java Basics - Anfänger-Themen 12
U Array.sort auf variable Array-Größe anwenden Java Basics - Anfänger-Themen 3
G float-Array _ohne_ Arrays.sort sortieren Java Basics - Anfänger-Themen 5
T Array verkleinern Java Basics - Anfänger-Themen 2
J Array aus Numberfield Eingaben Java Basics - Anfänger-Themen 7
D Array List mit Objekten sortieren Java Basics - Anfänger-Themen 2
onlyxlia Anzahl Random Zahlen mit Scanner abfragen und in Array speichern Java Basics - Anfänger-Themen 10
Ü Java Array - Buchstaben als Zahlen ausgeben Java Basics - Anfänger-Themen 22
Ü Zweidimensionales Array in der ersten Zeile deklarieren Java Basics - Anfänger-Themen 13
Thomas Uppe 2D Array Reihenfolge vermischen Java Basics - Anfänger-Themen 4
T array auslesen Java Basics - Anfänger-Themen 2
Nitrogames Variablen Variable aus JOptionPane Abfrage in Array einfügen Java Basics - Anfänger-Themen 4
moini Auf Array aus Superklasse zugreifen? Java Basics - Anfänger-Themen 2
J ArrayList in 2D-Array konvertieren. Java Basics - Anfänger-Themen 48
M NullPointerException: Cannot read the array length because "this.Kinder" is null Java Basics - Anfänger-Themen 1
P Wieso kann ich als Index für einen Array einen Char angeben? Java Basics - Anfänger-Themen 3
Finn_lol Fehlermeldung bei Schleife mit Array Java Basics - Anfänger-Themen 4
Proxy Chars vor array übergabe toLowerUpcase Java Basics - Anfänger-Themen 14
iAmFaiinez Primzahlen Tester ohne Array Java Basics - Anfänger-Themen 4
S array 2 dimensional treppe Java Basics - Anfänger-Themen 3
S Array 2x2 Blöcke mit 0 und 1 Java Basics - Anfänger-Themen 10
C Array von Klassen Java Basics - Anfänger-Themen 2
julian0507 2Dim-Array Spaltensummen Java Basics - Anfänger-Themen 1
XWing Doppelte Zahlen im Array Java Basics - Anfänger-Themen 8
melisax Java 2D-Array Tabelle Java Basics - Anfänger-Themen 4
melisax Java Array Wert an bestimmtem Index angeben Java Basics - Anfänger-Themen 14
W Items löschen aus String Array vom Custom Base Adapter Java Basics - Anfänger-Themen 2
Proxy Stack erweitern mit neuem Array falls der alte voll ist!? Java Basics - Anfänger-Themen 5
E Array, nächste Zahl zur 5 ausgeben, wie? Java Basics - Anfänger-Themen 42
J Array.list vergleichen Java Basics - Anfänger-Themen 1
W Java-Code mit Array Java Basics - Anfänger-Themen 14
D Reflections & Generisches Array Java Basics - Anfänger-Themen 4
T Array Java Basics - Anfänger-Themen 2
T Array Java Basics - Anfänger-Themen 15
T Wörteranzahl im Array zählen Java Basics - Anfänger-Themen 9
Ostkreuz Zweidimensionaler Array Index Java Basics - Anfänger-Themen 2
S String Array Buchstaben um einen gewissen Wert verschieben Java Basics - Anfänger-Themen 4
R Images aus einem Array ausgeben Java Basics - Anfänger-Themen 3
R 2d Array individuell machen Java Basics - Anfänger-Themen 4
D 2D Char Array into String Java Basics - Anfänger-Themen 2
J Array Median bestimmen Java Basics - Anfänger-Themen 6
S Array Maximum bestimmen mit for und foreach Java Basics - Anfänger-Themen 7
S Prüfen ob ein zweidimensionales Array rechteckig ist Java Basics - Anfänger-Themen 4
N Array Java Basics - Anfänger-Themen 1
J Array Mittleren Wert bestimmen Java Basics - Anfänger-Themen 2
D OOP Array einem Objekt zuweisen Java Basics - Anfänger-Themen 2
O Zahlen aus einem char-array per char + Zeichen addieren Java Basics - Anfänger-Themen 2
S leeres Array statt Null Pointer Exception ausgeben Java Basics - Anfänger-Themen 20
S Inhalte aus Array vergleichen und Max ausgeben Java Basics - Anfänger-Themen 3
M 2d array ohne längen anlegen Java Basics - Anfänger-Themen 4
S Bestimmte werte aus einem Array löschen Java Basics - Anfänger-Themen 2
S Ausgeben wie oft ein Wert in einem Array vorkommt Java Basics - Anfänger-Themen 7
E Reihenfolge der Werte umdrehen (mittels statischem int-Array Java Basics - Anfänger-Themen 3
O 2 Dimensionales Array Java Basics - Anfänger-Themen 6
javaBoon86 Array mehrere Dimensionen Java Basics - Anfänger-Themen 10
B Array nach Elementwerten sortieren? Java Basics - Anfänger-Themen 1
B Explizit Array definieren geht nicht? Java Basics - Anfänger-Themen 14
D Kleinste Zahl in Array finden die vorher noch errechnet werden müssen. Java Basics - Anfänger-Themen 4
L Gegebenes Array sortieren, indem zufällige Zahlenpaare aus Array ausgewählt werden Java Basics - Anfänger-Themen 14
Say 2-DIM Array Code lesen und verstehen Java Basics - Anfänger-Themen 5
N Array beim erstellen mit Werten füllen Java Basics - Anfänger-Themen 6
C Java Array Struktur, welche ist wann besser? Java Basics - Anfänger-Themen 12
Temsky34 Array IndexOf nicht verfügbar Java Basics - Anfänger-Themen 18
belana wie am besten 2D Array von String to Integer Java Basics - Anfänger-Themen 18
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
E Durch Muster in Array iterieren Java Basics - Anfänger-Themen 3
L Frage zum Array Java Basics - Anfänger-Themen 1
C 2D Array Ausgabe mit for-Schleife i,j Java Basics - Anfänger-Themen 4
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
P Array vom Typ Klasse Java Basics - Anfänger-Themen 18
Lion.King Array deklarieren und initialisieren Java Basics - Anfänger-Themen 5
P Array-Objekte-Aufruf Java Basics - Anfänger-Themen 22
A CSv.Datei einlesen und die werte in zweidemosional Int Array speichern Java Basics - Anfänger-Themen 9
M Methoden Zweidimensionaler Array mit Setter Methode ändern Java Basics - Anfänger-Themen 4
AkiJou Zeile in 2d Array löschen Java Basics - Anfänger-Themen 2
LilliCherry Array in einer Zeile ausgeben Java Basics - Anfänger-Themen 6
A Elemente in einem Array Java Basics - Anfänger-Themen 5
A Vorkommende Farben ermittel und als Array zurückgeben Java Basics - Anfänger-Themen 7
AhmadSlack Array Java Basics - Anfänger-Themen 7
Jambolo Kartenhand Array Java Basics - Anfänger-Themen 14
ravenz Schleife mit for über String Array „zahlen“und prüfen ob Wert „a“ oder „b“ oder „c“ entspricht (mittels || ) Java Basics - Anfänger-Themen 4
S Eine Variable in einem Array speichern Java Basics - Anfänger-Themen 5
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
T String Array Fehler beim Index Java Basics - Anfänger-Themen 3
krgewb byte-Array, das ein Bild repräsentiert Java Basics - Anfänger-Themen 1
1 Array rekursiv durchlaufen Java Basics - Anfänger-Themen 8

Ähnliche Java Themen

Neue Themen


Oben