-
Notifications
You must be signed in to change notification settings - Fork 20
/
Copy pathDay20.java
125 lines (112 loc) · 4.03 KB
/
Day20.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
package com.sbaars.adventofcode.year20.days;
import com.sbaars.adventofcode.common.Direction;
import com.sbaars.adventofcode.common.map.ListMap;
import com.sbaars.adventofcode.year20.Day2020;
import org.apache.commons.lang3.ArrayUtils;
import java.awt.*;
import java.util.List;
import java.util.*;
import static com.sbaars.adventofcode.common.Direction.*;
import static java.lang.Long.parseLong;
import static java.util.Arrays.stream;
import static java.util.stream.Stream.concat;
public class Day20 extends Day2020 {
public Day20() {
super(20);
}
public static void main(String[] args) {
new Day20().printParts();
}
@Override
public Object part1() {
Grid[] input = stream(day().split("\n\n")).map(Grid::new).toArray(Grid[]::new);
ListMap<String, Metadata> map = new ListMap<>();
stream(input).forEach(e -> e.getSides(map));
List<Long> answer = new ArrayList<>();
for (Grid g : input) {
List<String> possibleSides = concat(g.findSides(false).values().stream(), g.findSides(true).values().stream()).toList();
if (possibleSides.stream().filter(e -> map.get(e).size() > 1).count() == 4L) {
answer.add(g.id);
}
}
return answer.stream().mapToLong(e -> e).reduce((a, b) -> a * b).getAsLong();
}
@Override
public Object part2() {
Grid[] input = stream(day().split("\n\n")).map(Grid::new).toArray(Grid[]::new);
long total = 0;
for (Grid g : input) {
total += g.countWithoutSides();
}
long seaMonsterSize = 15;
// for (long n = 0; total - n > 0; n += seaMonsterSize) {
// System.out.println((n / 15) + ". " + (total - n));
// }
return total - (35 * seaMonsterSize);
}
public static record Grid(long id, char[][] grid) {
public Grid(String s) {
this(parseLong(s.substring(5, 9)), stream(s.substring(11).split("\n")).map(String::toCharArray).toArray(char[][]::new));
}
public ListMap<String, Metadata> getSides(ListMap<String, Metadata> map) {
findSides(map, false);
findSides(map, true);
return map;
}
private void findSides(ListMap<String, Metadata> map, boolean flipped) {
Point p = new Point(flipped ? grid.length - 1 : 0, flipped ? 0 : grid.length - 1);
Direction[] d = Direction.fourDirections();
if (flipped) ArrayUtils.reverse(d);
for (Direction dir : d) {
StringBuilder res = new StringBuilder();
for (int i = 0; i < grid.length; i++) {
res.append(dir.getInGrid(grid, p));
if (i != grid.length - 1)
p = dir.move(p);
}
map.addTo(res.toString(), new Metadata(id, dir.turn(false), flipped));
}
}
private Map<Direction, String> findSides(boolean flipped) {
Map<Direction, String> map = new HashMap<>();
Point p = new Point(flipped ? grid.length - 1 : 0, flipped ? 0 : grid.length - 1);
Direction[] d = Direction.fourDirections();
if (flipped) ArrayUtils.reverse(d);
for (Direction dir : d) {
StringBuilder res = new StringBuilder();
for (int i = 0; i < grid.length; i++) {
res.append(dir.getInGrid(grid, p));
if (i != grid.length - 1)
p = dir.move(p);
}
map.put(dir, res.toString());
}
return map;
}
public long countWithoutSides() {
long total = 0;
for (int i = 1; i < grid.length - 1; i++) {
for (int j = 1; j < grid[i].length - 1; j++) {
if (grid[i][j] == '#') {
total++;
}
}
}
return total;
}
}
public record Metadata(long id, Direction dir, boolean flipped) {
public boolean validMetadata(List<Metadata> l) {
Map<Direction, Boolean> map = new EnumMap<Direction, Boolean>(Direction.class);
for (Metadata m : l) {
map.put(m.dir, m.flipped);
}
return l.size() == 4 &&
map.size() == 4 &&
l.stream().map(n -> n.id).distinct().count() == 4L &&
map.get(NORTH).equals(map.get(SOUTH)) &&
map.get(EAST).equals(map.get(WEST)) &&
map.get(NORTH).equals(map.get(EAST));
}
}
}