Skip to content

Commit 0bdbd8a

Browse files
committed
binary-search helper fn
1 parent 08a28ef commit 0bdbd8a

File tree

4 files changed

+75
-13
lines changed

4 files changed

+75
-13
lines changed

notebooks/y2024/d18.clj

+4-13
Original file line numberDiff line numberDiff line change
@@ -105,21 +105,12 @@
105105
; Dette løser vi ved å snevre inn søkeområdet basert på om vi finner en gyldig path på
106106
; starten, midten og slutten av søkeområdet.
107107

108-
; (*TODO: denne algoritmen trenger litt refaktorering*)
109-
110108
(defn part-2 [dimensions input]
111-
(let [num-bytes (alength (.split input "\n"))
112-
corrupted-locations (corrupted-locations input)]
109+
(let [total-number-of-bytes (alength (.split input "\n"))
110+
corrupted-locations (corrupted-locations input)
111+
has-valid-path? #(find-path dimensions (set (take % corrupted-locations)))]
113112
(->> (nth corrupted-locations
114-
(loop [[start end] [0 num-bytes]]
115-
(let [middle (int (+ (/ (- end start) 2) start))
116-
results (->> [start middle end]
117-
(map #(some? (find-path dimensions (set (take % corrupted-locations))))))
118-
index-result (zipmap [start middle end] results)]
119-
(if (= 2 (count index-result))
120-
(first (keep (fn [[k v]] (when v k)) index-result))
121-
(recur [(last (filter index-result [start middle end]))
122-
(first (remove index-result [start middle end]))])))))
113+
(utils/binary-search has-valid-path? (dec total-number-of-bytes)))
123114
reverse
124115
(map str)
125116
(String/join ","))))

src/advent_of_code_clj/utils.clj

+25
Original file line numberDiff line numberDiff line change
@@ -74,3 +74,28 @@
7474
(when next-nodes
7575
(.addAll visited next-nodes))
7676
(seq next-nodes)))))))
77+
78+
(defn binary-search
79+
"Find the first or last value where a predicate is true in a range.
80+
81+
Tests the start, middle and end of the value range and narrows down
82+
the search space until a value is found"
83+
([pred end]
84+
(binary-search pred 0 end))
85+
([pred start end]
86+
(let [middle (quot (+ start end) 2)
87+
p-start (pred start)
88+
p-mid (pred middle)
89+
p-end (pred end)]
90+
(cond
91+
(= p-start p-mid p-end) (throw (ex-info "Could not find a solution: start, middle and end had same equality" {}))
92+
(= start middle) (if p-start start end)
93+
(and p-mid (not p-end)) (binary-search pred middle end)
94+
(and p-start (not p-mid)) (binary-search pred start middle)
95+
(and (not p-start) p-mid) (binary-search pred start middle)
96+
(and (not p-mid) p-end) (binary-search pred middle end)))))
97+
98+
(defn depth-search [graph-fn start-node]
99+
(let [visited (doto (HashSet.) (.add start-node))]
100+
(loop [node [start-node]]
101+
(let [neighbours (graph-fn node)]))))
+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
(ns advent-of-code-clj.utils-test
2+
(:require
3+
[advent-of-code-clj.utils :as utils]
4+
[clojure.test :refer [deftest is testing]]))
5+
6+
(deftest binary-search-test
7+
(testing "Finds last valid value"
8+
(let [numbers (vec (range 0 1000000 19))
9+
found-index (utils/binary-search #(> 100006 (numbers %)) 0 (dec (count numbers)))]
10+
(is (> (numbers (inc found-index)) 100006 (numbers found-index)))))
11+
(testing "Finds first valid value"
12+
(let [numbers (vec (range 0 1000000 19))
13+
found-index (utils/binary-search #(< 100006 (numbers %)) 0 (dec (count numbers)))]
14+
(is (< (numbers (dec found-index)) 100006 (numbers found-index)))))
15+
(testing "Inverted greater-than/less-than does not find same value"
16+
(let [numbers (vec (range 0 1000000 19))
17+
found-index (utils/binary-search #(< 100006 (numbers %)) 0 (dec (count numbers)))
18+
found-index-2 (utils/binary-search #(> 100006 (numbers %)) 0 (dec (count numbers)))]
19+
(is (not= found-index found-index-2)))
20+
(let [numbers (vec (range 0 1000000))
21+
middle-value (/ 1000000 2)
22+
found-index (utils/binary-search #(<= middle-value (numbers %)) 0 (dec (count numbers)))
23+
found-index-2 (utils/binary-search #(> middle-value (numbers %)) 0 (dec (count numbers)))]
24+
(is (= 500000 found-index))
25+
(is (= 499999 found-index-2))
26+
(is (not= found-index found-index-2))))
27+
(testing "Throws when no value can be found"
28+
(let [numbers (vec (range 0 1000000 19))]
29+
(is (thrown? Exception (utils/binary-search #(= 199 (numbers %)) 0 (dec (count numbers))))))
30+
(let [numbers (vec (repeat 1000 0))]
31+
(is (thrown? Exception (utils/binary-search #(= 0 (numbers %)) 0 (dec (count numbers))))))
32+
(let [numbers (vec (concat (range 0 400) (range 1000 0) (range 0 400)))]
33+
(is (thrown? Exception (utils/binary-search #(> (numbers %) 500) 0 (dec (count numbers))))))))

test/y2024/d18_test.clj

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
(ns y2024.d18-test
2+
(:require
3+
[advent-of-code-clj.input :as input]
4+
[clojure.test :refer [deftest is]]
5+
[y2024.d18 :as sut]))
6+
7+
(deftest part-1-test
8+
(is (= 22 (sut/part-1 [6 6] 12 sut/test-input)))
9+
(is (= 264 (sut/part-1 [70 70] 1024 (input/get-input 2024 18)))))
10+
11+
(deftest part-2-test
12+
(is (= "6,1" (sut/part-2 [6 6] sut/test-input)))
13+
(is (= "41,26" (sut/part-2 [70 70] (input/get-input 2024 18)))))

0 commit comments

Comments
 (0)