Frage zu Java Streams min / max

8u3631984

Bekanntes Mitglied
Ich habe eine Frage zur min und max Funktion in Java Streams :
Nehmen wir einmal an ich habe eine Liste mit Integers :

Java:
        List<Integer> numbers = new ArrayList<>();
        numbers.add(31);
        numbers.add(35);
        numbers.add(32);
        numbers.add(41);
        numbers.add(45);
        numbers.add(42);
        numbers.add(21);
        numbers.add(25);
        numbers.add(22);
        numbers.add(11);
        numbers.add(15);

Wenn ich nun die min oder max Funktion aufrufe, benötige ich einen Comperator :
Code:
  numbers.stream().min(Comparator.naturalOrder()).ifPresent(System.out::println);

MIr ist der Vorteil hier nicht ganz klar :
IOch könnte die Liste ja auch normnal sorieren und dann das erste (min) bzw (letzte) Element zurücklioefern.

Noch eine Frage ohne es getestet zu haben :
Wenn ich die Liste umgekehrt sortiere (also das größte Element zuerst) und dann min / max aufrufe, bekomme ich dann einen anderen Wert als mit der natürlichen Ordnung ?
 

LimDul

Top Contributor
MIr ist der Vorteil hier nicht ganz klar :
IOch könnte die Liste ja auch normnal sorieren und dann das erste (min) bzw (letzte) Element zurücklioefern.
Die Vorteile sind:
  • Kürzer
  • Lesbarer

Vor allem das lesbarer finde ich wichtig. So nimmt man beim lesen des Source Codes direkt bewusst war, dass hier das größte bzw. kleinste Element gesucht wird. Bei einer Sortierung und dann findFirst/findLast ist das nicht mehr direkt offensichtlich, sondern man muss da die Transferleistung erbringen, dass man sieht "Sortierung + erstes Element = kleinstes"

Hinzu kommt, dass je nach Implementierung min/max schneller sein kann. Min bzw. Max kann ich in O(n) implementieren, eine Sortierung braucht O(n*log n)
 

Oneixee5

Top Contributor
Den Ausdruck:
Java:
numbers.stream().min(Comparator.naturalOrder()).ifPresent(System.out::println);
kann man auch ersetzen durch:
Java:
List.of(1, 2, 34, 5, -1).stream().reduce(Math::min).ifPresent(System.out::println);
Dann benötigt keinen Comparator. Es ist aber notwendig irgendwie die Listenelemente zu vergleichen. Man wählt eben, die für den Programmierer, sprechendste Variante.
 

mihe7

Top Contributor
MIr ist der Vorteil hier nicht ganz klar :
Mir ist die Frage nicht ganz klar: von welchem Vorteil sprichst Du? Vom Stream oder davon, einen Comparator zu verwenden?

Wenn ich die Liste umgekehrt sortiere (also das größte Element zuerst) und dann min / max aufrufe, bekomme ich dann einen anderen Wert als mit der natürlichen Ordnung ?
Jein. Der Comparator definiert eine Ordnung und entsprechend dieser Ordnung gibt es in einer endlichen Menge ein kleinstes und ein größtes Element - es kann aber je nach Implementierung und Daten auch mehrere kleinste/größte Elemente geben.

Nehmen wir mal an, Du hast Personen und die haben ein Alter. Es kann aber natürlich mehrere Personen mit dem gleichen Alter geben. Ein Comparator, der die Ordnung der Personen nur über das Alter definiert, führt also daztu, dass mehrere Personen die kleinsten Elemente darstellen. Beim Finden des Minimums kann dann die Reihenfolge ausschlaggebend sein, in der die Elemente durchsucht werden (theoretisch kann auch zufällig ein Element aus der Menge der "kleinsten Personen" gewählt werden).

Existiert aber tatsächlich nur ein einziges kleinstes Element, dann ist die Reihenfolge, in der die Elemente durchsucht werden völlig egal.

Java:
public Optional<Person> juengste(List<Person> personen) {
    if (personen.isEmpty()) return Optional.empty(); 

    Person juengsteBisher = personen.get(0);
    for (int i = 1; i < personen.size(); i++) {
        Person person = personen.get(i);
        if (alterComparator.compare(person, juengsteBisher) < 0) {
            juengsteBisher = person;
        }
    }

    return Optional.of(juengsteBisher);
}
Bei dieser Implementierung würde die jüngste Person zurückgegeben, die vor allen anderen gleichaltrigen Personen in der Liste steht. Ersetzt Du die if-Bedingung < 0 durch <= 0, dann würdest Du die jüngste Person erhalten, die nach allen anderen gleichaltrigen Personen in der Liste steht. Dementsprechend hat die Reihenfolge der Personen in der Liste einen Einfluss auf das Ergebnis.

Gibt es allerdings nur eine jüngste Person (keine weiteren gleichaltrigen Personen), dann ist das Ergebnis eindeutig und unabhängig von der Reihenfolge in der Liste.
 

distinct

Mitglied
Zur Sicherheit noch die viel besser lesbare Stream-Variante:

Java:
import java.util.Arrays;
import java.util.Comparator;

public class Searching {
    record AlterUndPerson(int alter, String name) {
    }

    public static void main(String[] args) {
        AlterUndPerson[] alterUndPersonArray = {
                new AlterUndPerson(8, "Hänschen"),
                new AlterUndPerson(5, "Phillip"),
                new AlterUndPerson(99, "Gerda"),
                new AlterUndPerson(110, "Uroma"),
                new AlterUndPerson(14, "Donrat"),
                new AlterUndPerson(13, "Zzz"),
                new AlterUndPerson(8, "Miha"),
                new AlterUndPerson(5, "Pia"),
                new AlterUndPerson(5, "Ia"),
                new AlterUndPerson(10, "Hans")
        };
        Arrays.stream(alterUndPersonArray)
                .filter(aup -> aup.alter == Arrays.stream(alterUndPersonArray)
                        .min(Comparator.comparingInt(AlterUndPerson::alter))
                        .orElseThrow().alter)
                .forEach(System.out::println);
    }
}
 

Oneixee5

Top Contributor
Zur Sicherheit noch die viel besser lesbare Stream-Variante:

Java:
import java.util.Arrays;
import java.util.Comparator;

public class Searching {
    record AlterUndPerson(int alter, String name) {
    }

    public static void main(String[] args) {
        AlterUndPerson[] alterUndPersonArray = {
                new AlterUndPerson(8, "Hänschen"),
                new AlterUndPerson(5, "Phillip"),
                new AlterUndPerson(99, "Gerda"),
                new AlterUndPerson(110, "Uroma"),
                new AlterUndPerson(14, "Donrat"),
                new AlterUndPerson(13, "Zzz"),
                new AlterUndPerson(8, "Miha"),
                new AlterUndPerson(5, "Pia"),
                new AlterUndPerson(5, "Ia"),
                new AlterUndPerson(10, "Hans")
        };
        Arrays.stream(alterUndPersonArray)
                .filter(aup -> aup.alter == Arrays.stream(alterUndPersonArray)
                        .min(Comparator.comparingInt(AlterUndPerson::alter))
                        .orElseThrow().alter)
                .forEach(System.out::println);
    }
}
Aber das ist doch ein Joke 🤦‍♂️
 

Oneixee5

Top Contributor
Zur Sicherheit noch die viel besser lesbare Stream-Variante:

Java:
import java.util.Arrays;
import java.util.Comparator;

public class Searching {
    record AlterUndPerson(int alter, String name) {
    }

    public static void main(String[] args) {
        AlterUndPerson[] alterUndPersonArray = {
                new AlterUndPerson(8, "Hänschen"),
                new AlterUndPerson(5, "Phillip"),
                new AlterUndPerson(99, "Gerda"),
                new AlterUndPerson(110, "Uroma"),
                new AlterUndPerson(14, "Donrat"),
                new AlterUndPerson(13, "Zzz"),
                new AlterUndPerson(8, "Miha"),
                new AlterUndPerson(5, "Pia"),
                new AlterUndPerson(5, "Ia"),
                new AlterUndPerson(10, "Hans")
        };
        Arrays.stream(alterUndPersonArray)
                .filter(aup -> aup.alter == Arrays.stream(alterUndPersonArray)
                        .min(Comparator.comparingInt(AlterUndPerson::alter))
                        .orElseThrow().alter)
                .forEach(System.out::println);
    }
}

Wenn es unbedingt in einem Stream gelöst werden soll:
Java:
public class Main {

    record AlterUndPerson(int alter, String name) {
    }

    public static void main(final String[] args) {
        final AlterUndPerson[] alterUndPersonArray = {
                new AlterUndPerson(8, "Hänschen"),
                new AlterUndPerson(5, "Phillip"),
                new AlterUndPerson(99, "Gerda"),
                new AlterUndPerson(110, "Uroma"),
                new AlterUndPerson(14, "Donrat"),
                new AlterUndPerson(13, "Zzz"),
                new AlterUndPerson(8, "Miha"),
                new AlterUndPerson(5, "Pia"),
                new AlterUndPerson(5, "Ia"),
                new AlterUndPerson(10, "Hans")
        };

        Arrays.stream(alterUndPersonArray)
            .collect(Collectors.groupingBy(AlterUndPerson::alter))
            .entrySet()
            .stream()
            .min(Comparator.comparing(Entry::getKey))
            .get()
            .getValue()
            .forEach(System.out::println);
    }

}
 

distinct

Mitglied
jup, sieht besser aus, weil der innere Stream vermiden wird.

noch besser wäre es, vor dem Stream das Min. zu speichern. Aber dann wäre es nicht mehr funktional.
 

Oneixee5

Top Contributor
MIr ist der Vorteil hier nicht ganz klar :
Um noch mal auf den Vorteil zu kommen. Ich würde es ohne Stream lösen:
Java:
        final var numbers = new ArrayList<Integer>();
            numbers.add(31);
            numbers.add(35);
            numbers.add(32);
            numbers.add(41);
            numbers.add(45);
            numbers.add(42);
            numbers.add(21);
            numbers.add(25);
            numbers.add(22);
            numbers.add(11);
            numbers.add(15);

        System.out.format("min: %s%n", Collections.min(numbers));
Einen Vorteil kann ich hier auch mit Comparator nicht erkennen, da Integer bereits Comparable<Integer> implementiert.
Java:
        System.out.format("min: %s%n", Collections.min(numbers, Comparator.naturalOrder()));
 
Zuletzt bearbeitet:

mihe7

Top Contributor
Java:
IntSummaryStatistics stats = IntStream.of(31, 35, 32, 41, 45, 42, 21, 25, 22, 11, 15).summaryStatistics();
System.out.format("min: %d, max: %d%n", stats.getMin(), stats.getMax());
 

distinct

Mitglied
Dennoch, erst Grüppchen bilden, dann das minimalste Grüppchen bestimmen, dann das minimalste Grüppchen ausgeben lassen, sollte der richtigere Ansatz im Sinne der funktionalen Programmierung sein. :)

Nicht das langsame, hybride Konstrukt von mir ...

Wenn du eine rauchen möchtest, aber dein Feuerzeug ganz leer ist, würdest du auch jmd nach Feuer fragen, nicht nach Feuerzeuggas zum Nachfüllen, da Ersteres wahrscheinlicher ist, denke ich mal.
 

mihe7

Top Contributor
Dennoch, erst Grüppchen bilden, dann das minimalste Grüppchen bestimmen, dann das minimalste Grüppchen ausgeben lassen, sollte der richtigere Ansatz im Sinne der funktionalen Programmierung sein. :)
Und ich dachte, die Cannabislegalisierung wäre noch nicht ganz durch.

Java:
Arrays.stream(alterUndPersonArray).min(Comparator.comparingInt(AlterUndPerson::alter)).ifPresent(System.out::println);
 
Ähnliche Java Themen
  Titel Forum Antworten Datum
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
L Java frage Allgemeine Java-Themen 3
O Frage zum Runtimeverhalten von Java ... Allgemeine Java-Themen 2
M Allgemeine Frage: Wie lernt man Java / Programmieren von Grund auf? Allgemeine Java-Themen 7
A Methoden Allgemeine Java Frage Allgemeine Java-Themen 3
C J2V8 NodeJs Java Bride Problem und Frage!?!? Allgemeine Java-Themen 1
T Frage zu UML in Java programmieren Allgemeine Java-Themen 1
MTJ004 FTP Frage zu FTP Speicherung Java-Android-FTP Allgemeine Java-Themen 5
J Java Zufallsgenerator (6 aus 49) Frage Allgemeine Java-Themen 7
S Java Design Frage Allgemeine Java-Themen 10
D Classpath Frage zum Java Resource Loading Allgemeine Java-Themen 2
D Frage zu Java und Umlauten / charsets Allgemeine Java-Themen 2
B Frage zu Java und OpenGL? Allgemeine Java-Themen 3
H XML-File mit Java erzeugt Frage Allgemeine Java-Themen 10
H Java Vector Frage Allgemeine Java-Themen 9
H Frage zu java.weka.core.Instances Allgemeine Java-Themen 3
S Frage: Google Maps "Map" in Java Allgemeine Java-Themen 3
R Java Regex Frage Allgemeine Java-Themen 17
J Java-Datei unter Mac OS X öffnen - eine Frage der Klasse Allgemeine Java-Themen 2
T Frage zur Umsetzung in Java Allgemeine Java-Themen 4
S Frage zu verschiedenen Java Projekten Allgemeine Java-Themen 6
J MP3 mit Java - Frage zu JList Allgemeine Java-Themen 5
F Frage zu JSP / Java Programmierung Allgemeine Java-Themen 2
D Frage zu Java 7 Allgemeine Java-Themen 3
M Java Garbage Collector Frage (Singleton Pattern) Allgemeine Java-Themen 13
X Java 2D Spielfeld-Frage Allgemeine Java-Themen 4
K Frage zum thema Java und Internet Allgemeine Java-Themen 49
H Frage zu Wildcard Eingabe in Java! Allgemeine Java-Themen 5
M Eine Frage über Unit-Tests mit Java Allgemeine Java-Themen 2
S Frage zum Java-Konzept Allgemeine Java-Themen 6
S Frage zu Java und Plattformunabhängigkeit Allgemeine Java-Themen 2
N frage zu java konventionen Allgemeine Java-Themen 10
L Frage zu java.util.regex und Einlesen einer Datei Allgemeine Java-Themen 2
L Blöde Frage: Java für TOS? Allgemeine Java-Themen 6
G Eine Frage zu Java Allgemeine Java-Themen 15
T Frage zu Java-Class/Librarypath Allgemeine Java-Themen 2
S Frage zu jTDS, JAVA allgemein und Timer Allgemeine Java-Themen 6
V Frage zu Umlauten in Java Allgemeine Java-Themen 4
KonradN Mal eine Frage zu Binary Serialization Allgemeine Java-Themen 15
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
S Noch eine Design-Frage zu Setter Allgemeine Java-Themen 6
B For-Loop Frage Allgemeine Java-Themen 21
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
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
rentasad Design-Frage - Interfaces, Klassen, statische Methoden Allgemeine Java-Themen 3
S Frage zur JLS Allgemeine Java-Themen 0
J Verständnis Frage zur Instanz, Objekte, Instanzierung, Referenz Allgemeine Java-Themen 14
E String Frage Allgemeine Java-Themen 9
I bin neu bei GitHub, Frage zur Sicherheit Allgemeine Java-Themen 14
C KeyListener Frage Allgemeine Java-Themen 3
R Konstanten initialisieren - FRAGE Allgemeine Java-Themen 3
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
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
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
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

Ähnliche Java Themen

Neue Themen


Oben