Dieser Tag ist wieder ziemlich einfach und lässt sich mit Rekursion straight-forward lösen.
Grundsätzliche Vorgehensweise: Starte pro 0-Höhe einen rekursiven Aufruf, der jeweils wiederum in alle 4 möglichen Richtungen einen weiteren rekursiven Aufruf startet.
Tipps zur Vorgehensweise:
Lösungsvorschlag:
private int width;
private int height;
// Speichert pro Startkoordinate alle gefundenen Enden
// Jeweils als eindimensionales Array mit index 0 = x-Koordinate und index 1 = y-Koordinate
private ArrayList<int[]> trailEnds;
public void partOne() {
// Instanzvariablen (!) für Höhe und Breite der Karte
width = inputLines.get(0).length();
height = inputLines.size();
// Karte als int-Array
int[][] map = new int[width][height];
// Speichere den Input als int-Array
for (int y = 0; y < height; y++) {
String line = inputLines.get(y);
for (int x = 0; x < width; x++) {
map[x][y] = (int)(line.charAt(x)-'0');
}
}
int sumOfTrails = 0; // speichert die Summe aller Trails
// Iteriere über jede Koordinate
for (int x = 0; x < width; x++) {
for (int y = 0; y < height; y++) {
// Wenn die Koordinate mit 0 beginnt, dann starte den rekursiven Aufruf
if (map[x][y] == 0) {
trailEnds = new ArrayList(); // setzte diese Instanzvariable für die aktuelle Startposition zurück
sumOfTrails += trail(map, 0, x, y);
}
}
}
System.out.println(sumOfTrails);
}
/**
* Läuft rekursiv jeden möglichen Weg von bestimmten Startpunkt 0 zu jeder möglichen 9.
*
* @param map Die Karte
* @param h Erwartete Höhe des aktuellen Iterationsschritts
* @param x aktuelle x-Koordinate
* @param y aktuelle y-Koordinate
* @return Die Anzahl der gefunden Trail-Enden.
*/
private int trail(int[][] map, int h, int x, int y) {
// Basisfall: Aktuelle Koordinate (x,y) liegt außerhalb der Map
if (x < 0 || x >= width || y < 0 || y >= height) {
return 0;
}
// Basisfall: Aktuelle Koordinate (x,y) hat nicht die erwartete Höhe/Wert
if (map[x][y] != h) {
return 0;
}
// Basisfall: Aktuelle Koordinate (x,y) ist bereits in den gefundenen Enden
// des aktuellen Startpunktes enthalten.
for (int[] end: trailEnds) {
if (end[0] == x && end[1] == y) {
return 0;
}
}
// (Erfolgreicher) Basisfall: Ein Ende des Trails wurde erreicht!
if (map[x][y] == 9) {
trailEnds.add(new int[]{x,y}); // Speichere das aktuelle Trail-Ende.
return 1; // return 1, damit dieses gefundene Trail-Ende gezählt wird.
}
// Zähle über alle 4 Richtungen von der aktuellen Koordinate die Summe aller gefundenen Trail-Enden
int sumOfTrails = 0;
sumOfTrails += trail(map, h+1, x+1, y);
sumOfTrails += trail(map, h+1, x-1, y);
sumOfTrails += trail(map, h+1, x, y+1);
sumOfTrails += trail(map, h+1, x, y-1);
return sumOfTrails;
}
Für den Teil 2 muss nur eine winzige Kleinigkeit angepasst werden.
In Teil 1 geht es darum, pro Startpunkt zu zählen, wie viele Trail-Enden man erreichen kann. Jedes Trail-Ende darf also nur einmal erreicht werden. Daher mussten wir uns auch merken, ob wir ein Ende bereits erreicht hatten.
In Teil 2 geht es darum, pro Startpunkt zu zählen, wie oft ein Trail-Ende erreicht werden kann. Jedes Trail-Ende darf also von "benachbarten" rekursiven Aufrufen beliebig oft erreicht werden.
Achte bei den Aufrufen der rekursiven Methoden darauf, dass du überall die korrekten, neuen Methoden für Teil 2 aufrufst.
Lösungsvorschlag
public void partTwo() {
// Instanzvariablen (!) für Höhe und Breite der Karte
width = inputLines.get(0).length();
height = inputLines.size();
// Karte als int-Array
int[][] map = new int[width][height];
// Speichere den Input als int-Array
for (int y = 0; y < height; y++) {
String line = inputLines.get(y);
for (int x = 0; x < width; x++) {
map[x][y] = (int)(line.charAt(x)-'0');
}
}
int sumOfTrails = 0; // Zählt die Summe aller gefundenen Trails
// Finde jeden Startpunkt eines Trails
for (int x = 0; x < width; x++) {
for (int y = 0; y < height; y++) {
if (map[x][y] == 0) {
sumOfTrails += trail2(map, 0, x, y); // Starte den rekursiven Aufruf - diesmal OHNE Kopie
}
}
}
System.out.println(sumOfTrails);
}
private int trail2(int[][] map, int h, int x, int y) {
// Basisfall: Aktuelle Koordinate (x,y) liegt außerhalb der Map
if (x < 0 || x >= width || y < 0 || y >= height) {
return 0;
}
// Basisfall: Aktuelle Koordinate (x,y) hat nicht die erwartete Höhe/Wert
if (map[x][y] != h) {
return 0;
}
// (Erfolgreicher) Basisfall: Ein Ende des Trails wurde erreicht!
if (map[x][y] == 9) {
return 1; // return 1, damit dieses gefundene Trail-Ende gezählt wird.
}
// Zähle über alle 4 Richtungen von der aktuellen Koordinate die Summe aller gefundenen Trail-Enden
int sumOfTrails = 0;
sumOfTrails += trail2(map, h+1, x+1, y);
sumOfTrails += trail2(map, h+1, x-1, y);
sumOfTrails += trail2(map, h+1, x, y+1);
sumOfTrails += trail2(map, h+1, x, y-1);
return sumOfTrails;
}