faecher:informatik:oberstufe:java:aoc:aco2023:day2:start

Tag 2

  • Du kannst/musst exzessiv die split("…")-Methode von Java nutzen. Damit zerteilst du einen String an jedem gefundenen Trennzeichen, sodass ein String-Array entsteht. Beispiel: line.split(":") erzeugt ein Array der Form ["Game 1", "4 red, 1 green, 15 blue; 6 green UND SO WEITER"]. Das Array enthält also zwei Strings: Erstens den Teil VOR dem Doppelpunkt, und zweitens den gesamten restlichen Teil hinter dem Doppelpunkt.
  • Die split("…")-Methode kannst du mehrmals direkt hintereinander aufrufen und zwischendurch mit dem üblichen Index-Zugriff in eckigen Klammern auf einen bestimmten String des String-Arrays zugreifen. Beispiel: Der Befehl line.split(":")[1].split(";") gibt dir ein String Array, bei dem jeder String ein Herausziehen des Elfen darstellt. Das Ergebnis ist dann also z. B. die Form ["4 red, 1 green, 15 blue", "6 green, 2 red, 10 blue", "…"]
  • Unterteile jeden dieser Strings noch einmal mehr (an einem weiteren Satzzeichen), um jede einzelne Farbe zu erhalten.
  • Nun kannst du für eine einzelne Farbe (z. B. 3 green) prüfen, ob diese Anzahl für diese Farbe erlaubt ist. Dazu kannst du den String ein weiteres Mal am Leerzeichen zerteilen und für den String-Vergleich equals() benutzen.
  • Es empfiehlt sich immer wieder vorsorglich die strip() Methode zu nutzen. Diese entfernt Whitespace-Character (also z. B. ein Leerzeichen) am Anfang eines Strings und am Ende.
  • Nutze eine boolean-Variable, um festzuhalten, ob du in der aktuellen Zeile bereits eine illegale Anzahl einer Farbe gezogen hast. Dann kannst du nämlich früher die Schleifen unterbrechen (break;) und weißt am Ende auch, ob denn die aktuelle Zeilennummer aufaddiert werden muss oder nicht.

Lösungsvorschlag Teil 1

  • Das meiste der verschachtelten Schleifen kannst du stehen lassen. Du benötigst nun nicht mehr die boolean-Variable, um zu überprüfen, ob die aktuelle Zeile 'erlaubt' ist. Jede Zeile fließt gleichermaßen in das Ergebnis ein.
  • Erstelle pro Zeile drei int-Variablen, die dir das größte Vorkommen von rot, grün und blau Speichern. Für jede Farbe überprüfst du, ob die Zahl der jeweiligen Farbe größer ist als der bisher gespeicherte Wert in den neuen Variablen.
  • Am Ende jeder Zeile fügst du das Produkt der drei Variablen zur Gesamtsumme hinzu.

Lösungsvorschlag Teil 2

Das Prinzip von Variante 2 ist dasselbe wie bei Variante 1, die zentrale Methode zur Aufteilung der Zeilen ist und bleibt String.split(). Variante 2 modelliert das Problem objektorientiert:

Anmerkung: Das ist natürlich Overkill - vor allem für die Züge eine eigene Klasse zu implementieren, die letztlich nur die Werte für RGB enthält - aber es zeigt das Prinzip der Objektorientierung dennoch ganz gut:

  • Jede Klasse definiert einen neuen Datentyp, hier erhalten wir Spiel um Spiele mit ihren Zügen zu speichern, und Zug um einen Zug zu speichern.
  • Die Klasse day2 verwaltet eine ArrayList des Typs Spiel: private ArrayList<Spiel> spiele;
  • Die Klasse Spiel speichert alle Züge, die zu einem Spiel gehören in einer ArrayList des Typs Zug: private ArrayList<Zug> zuege;

Nun verlagert sich die Arbeit vor allem auf das Einlesen der Rätseldaten in der Methode getGames() - wenn man das hat, kann man so ziemlich alle Infos sehr einfach extrahieren.

getGames() macht prinzipiell folgendes:

  • Splitte die Eingabezeile am Doppelpunkt, der erste Teil beinhaltet die Infos zur Spielnummer, wenn man die Spielnummer extrahiert hat, kann man das neue Spiel erzeugen.
  • Der zweite Teil ist ein String mit allen Zügen zu diesem Spiel.

Tipp: Erster Split & Erzeugung des Spiel-Objekts

  • Die einzelnen Züge erhält man durch den Split des Strings mit allen Zügen am Semikolon, die einzelnen Farben dann, indem man die Substrings an den Kommata splittet:

Tipp: Gerüst mit Schleifen zu den Zugdetails

  • Wenn das Spiel dann komplett, muss es noch in die ArrayList eingefügt werden: spiele.add(s);

Die Lösungen für Teil 1 und 2 sehen in dieser Variante so aus:

Jetzt kann man direkt über die Spiele iterieren und alle Infos nach belieben auswerten.

public int partOne() {
        int answer=0;
        for(Spiel s: spiele) {
            if( s.getMax("red") <= redLimit &&
                s.getMax("green") <= greenLimit &&
                s.getMax("blue") <= blueLimit ) {
                    answer += s.getSpielNummer();
                }
        }
        return answer;
    }
 
    public int partTwo() {
        int answer=0;
        for(Spiel s: spiele) {
            int[] fewest = s.getFewestColorNums();
            answer += fewest[0] * fewest[1] * fewest[2];
        }
        return answer;
    }

Code zu dieser Variante: https://codeberg.org/qg-info-unterricht/aoc-bluej-2023 in den Dateien day2.java, Spiel.java und Zug.java.1)


  • faecher/informatik/oberstufe/java/aoc/aco2023/day2/start.txt
  • Zuletzt geändert: 06.12.2023 12:51
  • von Frank Schiebel