summary refs log tree commit diff homepage
diff options
context:
space:
mode:
authorJune McEnroe <june@causal.agency>2019-12-12 00:50:59 -0500
committerJune McEnroe <june@causal.agency>2020-11-22 00:14:26 -0500
commita5332be91a0aa565aedce2053f5bbc68dd470dbd (patch)
tree3ca8fc2af80bbbd164388e703ff8ef47f21aaead
parentSolve day 10 part 1 (diff)
downloadaoc-a5332be91a0aa565aedce2053f5bbc68dd470dbd.tar.gz
aoc-a5332be91a0aa565aedce2053f5bbc68dd470dbd.zip
Solve day 10 part 2
I'm so fucking frustrated with this one. I was stuck for hours trying to
sort the points by angle not realizing that (int)(angle(a) - angle(b))
would be 0 for all angles with differences smaller than 1. I also wanted
to do the sort by atan2(-y, x) - π/2 intuitively to get "up" be zero,
but I never got that to work.
-rw-r--r--2019/day10.c42
1 files changed, 40 insertions, 2 deletions
diff --git a/2019/day10.c b/2019/day10.c
index dbc2352..04743b4 100644
--- a/2019/day10.c
+++ b/2019/day10.c
@@ -1,3 +1,4 @@
+#include <math.h>
 #include <stdio.h>
 #include <stdlib.h>
 
@@ -25,6 +26,21 @@ static struct Point reduce(struct Point p) {
 	return p;
 }
 
+static double angle(struct Point p) {
+	double theta = atan2(p.x, -p.y);
+	return (theta < 0.0 ? theta + 2.0 * M_PI : theta);
+}
+
+static int compar(const void *_a, const void *_b) {
+	struct Point a = *(const struct Point *)_a;
+	struct Point b = *(const struct Point *)_b;
+	if (angle(a) == angle(b)) {
+		return copysign(1.0, hypot(a.x, a.y) - hypot(b.x, b.y));
+	} else {
+		return copysign(1.0, angle(a) - angle(b));
+	}
+}
+
 int main(void) {
 	struct Point ps[2048];
 	size_t len = 0;
@@ -38,7 +54,7 @@ int main(void) {
 		}
 	}
 
-	size_t max = 0;
+	size_t m, max = 0;
 	for (size_t i = 0; i < len; ++i) {
 		struct Point ds[2048];
 		size_t dlen = 0;
@@ -51,7 +67,29 @@ int main(void) {
 			}
 			if (d == dlen) ds[dlen++] = a;
 		}
-		if (dlen > max) max = dlen;
+		if (dlen > max) {
+			max = dlen;
+			m = i;
+		}
 	}
 	printf("%zu\n", max);
+
+	struct Point o = ps[m];
+	ps[m] = ps[--len];
+	for (size_t i = 0; i < len; ++i) {
+		ps[i] = sub(ps[i], o);
+	}
+	qsort(ps, len, sizeof(ps[0]), compar);
+	for (int n = 0;;) {
+		for (size_t i = 0; i < len; ++i) {
+			if (eq(ps[i], Point(0, 0))) continue;
+			if (++n == 200) {
+				printf("%d\n", 100 * (o.x + ps[i].x) + (o.y + ps[i].y));
+				return 0;
+			}
+			double theta = angle(ps[i]);
+			ps[i] = Point(0, 0);
+			while (i + 1 < len && angle(ps[i + 1]) == theta) i++;
+		}
+	}
 }
='2021-01-12 22:24:33 -0500'>2021-01-12Don't output a pre in hilex by defaultJune McEnroe 2021-01-12Move hilex out of hilex directoryJune McEnroe 2021-01-12Consolidate hilex formatters into hilex.cJune McEnroe 2021-01-12Remove hacky tagging from hilexJune McEnroe 2021-01-12Add htagml -iJune McEnroe 2021-01-12Render tag index in HTMLJune McEnroe 2021-01-12Add htagml -xJune McEnroe 2021-01-12Prevent matching the same tag twiceJune McEnroe 2021-01-12Process htagml file line by lineJune McEnroe 2021-01-12Split fields by tab onlyJune McEnroe 2021-01-12List both Makefile and html.sh under README.7June McEnroe 2021-01-12Add htagml exampleJune McEnroe 2021-01-12Use mandoc and htagml for bin htmlJune McEnroe 2021-01-12Add htagmlJune McEnroe 2021-01-12Replace causal.agency with a simple mdoc pageJune McEnroe 2021-01-11Publish "Using vi"June McEnroe 2021-01-11Enable diff.colorMovedJune McEnroe 2021-01-10Set less search case-insensitiveJune McEnroe 2021-01-10Set EXINITJune McEnroe 2021-01-09Add c -t flag to print expression typeJune McEnroe 2021-01-05Update taglineJune McEnroe