Skip to content

Commit 0a59b64

Browse files
committed
binary-search -> bisect
1 parent f5e988b commit 0a59b64

File tree

8 files changed

+53
-36
lines changed

8 files changed

+53
-36
lines changed

deps.edn

+2-1
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,8 @@
2929
net.mikera/vectorz-clj {:mvn/version "0.48.0"}
3030
org.scicloj/clay {:mvn/version "2-beta25"}
3131
org.scicloj/kindly {:mvn/version "4-beta14"}
32-
uncomplicate/fluokitten {:mvn/version "0.10.0"}}
32+
uncomplicate/fluokitten {:mvn/version "0.10.0"}
33+
metosin/malli {:mvn/version "0.17.0"}}
3334
:aliases
3435
{:profiler {:extra-deps {com.clojure-goes-fast/clj-async-profiler {:mvn/version "1.5.1"}}
3536
:jvm-opts ["-Djdk.attach.allowAttachSelf"]}

input/2022/input1.nippy

9.58 KB
Binary file not shown.

notebooks/y2022/d01.clj

+6-7
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,8 @@
55
{:title "Y2022 - Day 1"}}
66
(:require [advent-of-code-clj.utils :refer [emap sort- split-newline sum]]
77
[clojure.string :as str]
8-
[nextjournal.clerk :as clerk]))
8+
[nextjournal.clerk :as clerk]
9+
[advent-of-code-clj.input :as input]))
910

1011
^{::clerk/visibility {:code :hide}}
1112
(clerk/md "# Advent of code 2022, part 01")
@@ -54,20 +55,18 @@
5455
(part-1 test-data)
5556

5657
;; This yields the expected result, so we apply it to our input:
57-
^{::clerk/visibility {:code :hide}}
58-
(clerk/code
59-
'(= 66186 (part-1 (slurp "input/2022/01.txt"))))
58+
59+
(= 66186 (part-1 (input/get-input 2022 1)))
6060

6161
;; ## Part 2
6262
;; > find the amount of calories carried by the "top 3" elves
6363

6464
^{::clerk/visibility {:result :hide}}
6565
(defn part-2 [data]
6666
(->> data parse sorted-calories (take 3) sum))
67+
6768
(part-2 test-data)
6869

6970
;; Test output looks ok, so let's apply it to the read input
7071

71-
^{::clerk/visibility {:code :hide}}
72-
(clerk/code
73-
'(= 196804 (part-2 (slurp "input/2022/01.txt"))))
72+
(= 196804 (part-2 (input/get-input 2022 1)))

notebooks/y2024/d18.clj

+1-1
Original file line numberDiff line numberDiff line change
@@ -110,7 +110,7 @@
110110
corrupted-locations (corrupted-locations input)
111111
has-valid-path? #(find-path dimensions (set (take % corrupted-locations)))]
112112
(->> (nth corrupted-locations
113-
(utils/binary-search has-valid-path? (dec total-number-of-bytes)))
113+
(utils/bisect has-valid-path? (dec total-number-of-bytes)))
114114
reverse
115115
(map str)
116116
(String/join ","))))

src/advent_of_code_clj/utils.clj

+18-12
Original file line numberDiff line numberDiff line change
@@ -1,25 +1,30 @@
11
(ns advent-of-code-clj.utils
2-
(:require [clojure.string :as str])
32
(:import [java.util HashSet]))
43

5-
(defn coord-map [xs-of-xses]
4+
(defn coord-map
5+
{:malli/schema [:-> [:sequential [:sequential :any]] [:map-of [:tuple :int :int] :any]]}
6+
[xs-of-xses]
67
(->> xs-of-xses
78
(map-indexed (fn [idy xs]
89
(map-indexed (fn [idx v]
910
[[idx idy] v])
1011
xs)))
1112
(transduce cat merge)))
1213

13-
(defn coord-map-fixed [xs-of-xses]
14+
(defn coord-map-fixed
15+
{:malli/schema [:-> [:sequential [:sequential :any]] [:map-of [:tuple :int :int] :any]]}
16+
[xs-of-xses]
1417
(->> xs-of-xses
1518
(map-indexed (fn [idy xs]
1619
(map-indexed (fn [idx v]
1720
[[idy idx] v])
1821
xs)))
1922
(transduce cat merge)))
2023

21-
(defn text->matrix [text]
22-
(mapv vec (str/split-lines text)))
24+
(defn text->matrix
25+
{:malli/schema [:-> :string [:vector [:vector char?]]]}
26+
[text]
27+
(mapv vec (.split text "\n")))
2328

2429
(defn adjacent-hv
2530
"Find adjacent coordinates, without diagonals"
@@ -40,7 +45,7 @@
4045
[(dec x) (inc y)] [x (inc y)] [(inc x) (inc y)]])
4146

4247
(defn split-newline [text]
43-
(str/split text #"\n\n"))
48+
(.split text "\n\n"))
4449

4550
(defn emap [fun xs]
4651
(if (coll? xs)
@@ -75,13 +80,14 @@
7580
(.addAll visited next-nodes))
7681
(seq next-nodes)))))))
7782

78-
(defn binary-search
83+
(defn bisect
7984
"Find the first or last value where a predicate is true in a range.
8085
8186
Tests the start, middle and end of the value range and narrows down
8287
the search space until a value is found"
88+
{:malli/schema [:-> ifn? :int [:* :int] :int]}
8389
([pred end]
84-
(binary-search pred 0 end))
90+
(bisect pred 0 end))
8591
([pred start end]
8692
(let [middle (quot (+ start end) 2)
8793
p-start (pred start)
@@ -90,10 +96,10 @@
9096
(cond
9197
(= p-start p-mid p-end) (throw (ex-info "Could not find a solution: start, middle and end had same equality" {}))
9298
(= 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)))))
99+
(and p-mid (not p-end)) (bisect pred middle end)
100+
(and p-start (not p-mid)) (bisect pred start middle)
101+
(and (not p-start) p-mid) (bisect pred start middle)
102+
(and (not p-mid) p-end) (bisect pred middle end)))))
97103

98104
(defn depth-search [graph-fn start-node]
99105
(let [visited (doto (HashSet.) (.add start-node))]
+14-11
Original file line numberDiff line numberDiff line change
@@ -1,33 +1,36 @@
11
(ns advent-of-code-clj.utils-test
22
(:require
3+
[aoc.test-utils :as tu]
34
[advent-of-code-clj.utils :as utils]
4-
[clojure.test :refer [deftest is testing]]))
5+
[clojure.test :refer [deftest is testing use-fixtures]]))
56

6-
(deftest binary-search-test
7+
(use-fixtures :once tu/with-instrumentation)
8+
9+
(deftest bisect-test
710
(testing "Finds last valid value"
811
(let [numbers (vec (range 0 1000000 19))
9-
found-index (utils/binary-search #(> 100006 (numbers %)) 0 (dec (count numbers)))]
12+
found-index (utils/bisect #(> 100006 (numbers %)) 0 (dec (count numbers)))]
1013
(is (> (numbers (inc found-index)) 100006 (numbers found-index)))))
1114
(testing "Finds first valid value"
1215
(let [numbers (vec (range 0 1000000 19))
13-
found-index (utils/binary-search #(< 100006 (numbers %)) 0 (dec (count numbers)))]
16+
found-index (utils/bisect #(< 100006 (numbers %)) 0 (dec (count numbers)))]
1417
(is (< (numbers (dec found-index)) 100006 (numbers found-index)))))
1518
(testing "Inverted greater-than/less-than does not find same value"
1619
(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)))]
20+
found-index (utils/bisect #(< 100006 (numbers %)) 0 (dec (count numbers)))
21+
found-index-2 (utils/bisect #(> 100006 (numbers %)) 0 (dec (count numbers)))]
1922
(is (not= found-index found-index-2)))
2023
(let [numbers (vec (range 0 1000000))
2124
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)))]
25+
found-index (utils/bisect #(<= middle-value (numbers %)) 0 (dec (count numbers)))
26+
found-index-2 (utils/bisect #(> middle-value (numbers %)) 0 (dec (count numbers)))]
2427
(is (= 500000 found-index))
2528
(is (= 499999 found-index-2))
2629
(is (not= found-index found-index-2))))
2730
(testing "Throws when no value can be found"
2831
(let [numbers (vec (range 0 1000000 19))]
29-
(is (thrown? Exception (utils/binary-search #(= 199 (numbers %)) 0 (dec (count numbers))))))
32+
(is (thrown? Exception (utils/bisect #(= 199 (numbers %)) 0 (dec (count numbers))))))
3033
(let [numbers (vec (repeat 1000 0))]
31-
(is (thrown? Exception (utils/binary-search #(= 0 (numbers %)) 0 (dec (count numbers))))))
34+
(is (thrown? Exception (utils/bisect #(= 0 (numbers %)) 0 (dec (count numbers))))))
3235
(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))))))))
36+
(is (thrown? Exception (utils/bisect #(> (numbers %) 500) 0 (dec (count numbers))))))))

test/aoc/test_utils.clj

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
(ns aoc.test-utils
2+
(:require [malli.dev :as md]))
3+
4+
(defn with-instrumentation [f]
5+
(md/start!)
6+
(f)
7+
(md/stop!))

tests.edn

+5-4
Original file line numberDiff line numberDiff line change
@@ -11,8 +11,9 @@
1111
:kaocha.plugin.randomize/seed 156086537,
1212
:kaocha.plugin.randomize/randomize? true,
1313
:kaocha/plugins
14-
[:kaocha.plugin/randomize
15-
:kaocha.plugin/filter
16-
:kaocha.plugin/capture-output],
17-
:kaocha.plugin.capture-output/capture-output? true,
14+
[:randomize
15+
:filter
16+
:capture-output
17+
:profiling],
18+
:kaocha.plugin.capture-output/capture-output? true
1819
:kaocha/reporter [kaocha.report/dots]}

0 commit comments

Comments
 (0)