Collections: Manipulation
Manipulating Clojure’s Collections
Code
(ns collections-manipulation.core)
;; Map
(map (fn [x] (+ x 1)) '(1 2 3 4 5))
;; => (2 3 4 5 6)
(map #(+ % 1) [1 2 3 4 5])
;; => (2 3 4 5 6)
(map inc #{1 2 3 4 5})
;; => (2 5 4 3 6)
(map (fn [x] (val x)) {:a 1 :b 2 :c 3 :d 4 :e 5})
;; => (1 2 3 4 5)
;; Filter
(filter (fn [x] (even? x)) '(1 2 3 4 5))
;; => (2 4)
(filter #(even? %) [1 2 3 4 5])
;; => (2 4)
(filter even? #{1 2 3 4 5})
;; => (4 2)
(filter (fn [x] (even? (val x))) {:a 1 :b 2 :c 3})
;; => ([:b 2])
;; Reduce
(reduce (fn [acc x] (str acc x)) "" ["a" "b" "c"])
;; => "abc"
(reduce #(str %1 %2) "" ["a" "b" "c"])
;; => "abc"
(reduce str "" #{"a" "b" "c" "d"})
;; => "dabc"
(reduce (fn [x y] (str x (val y))) "" {:a "a" :b "b" :c "c"})
;; => "abc"
;; Side effects
(run! println [1 2 3 4 5])
;; => nil
(run! println '(1 2 3 4 5))
;; => nil
(run! println #{1 2 3 4 5})
;; => nil
(run! #(println (val %)) {:a 1 :b 2 :c 3 :d 4 :e 5})
;; => nil
(doseq [x [1 2 3 4 5]]
(println x))
;; => nil
(doseq [x '(1 2 3 4 5)]
(println x))
;; => nil
(doseq [x #{1 2 3 4 5}]
(println x))
;; => nil
(doseq [x {:a 1 :b 2 :c 3 :d 4 :e 5}]
(println (val x)))
;; => nil
;; Immutability
(def a-vector [1 2 3 4 5])
(def a-list '(1 2 3 4 5))
(def a-set #{1 2 3 4 5})
(def a-map {:a 1 :b 2 :c 3 :d 4 :e 5})
(map inc a-vector)
;; => (2 3 4 5 6)
a-vector
;; => [1 2 3 4 5]
(filter even? a-list)
;; => (2 4)
a-list
;; => (1 2 3 4 5)
a-set
;; => #{1 4 3 2 5}
(reduce str "" a-set)
;; => "14325"
a-set
;; => #{1 4 3 2 5}
(run! #(println (val %)) a-map)
;; => nil
a-map
;; => {:a 1, :b 2, :c 3, :d 4, :e 5}
(doseq [x a-map]
(println (val x)))
;; => nil
a-map
;; => {:a 1, :b 2, :c 3, :d 4, :e 5}
Explanation
Here’s a breakdown of the Clojure code:
- Namespace:
(ns collections-manipulation.core)
is declaring the namespace for the code that follows. - Map: Map applies a function to every item in a collection and returns a new collection with the results.
(map (fn (+ x 1)) '(1 2 3 4 5))
: This is mapping an anonymous function that adds 1 to each element of the list.(map #(+ % 1) [1 2 3 4 5])
: This does the same thing, but uses a shorter syntax for the anonymous function.(map inc #{1 2 3 4 5})
: Theinc
function increments its argument by 1, so this is equivalent to the previous examples.(map (fn (val x)) {:a 1 :b 2 :c 3 :d 4 :e 5})
: This maps theval
function (which retrieves the value from a key-value pair) over a map, effectively returning the values.
- Filter: Filter applies a predicate function to each item in a collection and returns a new collection with only the items where the predicate returns true.
(filter (fn (even? x)) '(1 2 3 4 5))
: Filters the list to only include even numbers.(filter #(even? %) [1 2 3 4 5])
: Does the same thing using a shorter syntax.(filter even? #{1 2 3 4 5})
: Equivalent to the previous examples.(filter (fn (even? (val x))) {:a 1 :b 2 :c 3})
: Filters the map to only include key-value pairs where the value is even.
- Reduce: Reduce applies a binary function to a start value and each item in a collection in sequence, and returns a single value that aggregates the result.
(reduce (fn [acc x] (str acc x)) "" ["a" "b" "c"])
: Concatenates the strings in the vector.(reduce #(str %1 %2) "" ["a" "b" "c"])
: Does the same thing using a shorter syntax.(reduce str "" #{"a" "b" "c" "d"})
: Concatenates the strings in the set.(reduce (fn [x y] (str x (val y))) "" {:a "a" :b "b" :c "c"})
: Concatenates the values in the map.
- Side Effects:
run!
anddoseq
are used for side effects. They do not return meaningful values.(run! println [1 2 3 4 5])
: Prints each element in the vector.(run! #(println (val %)) {:a 1 :b 2 :c 3 :d 4 :e 5})
: Prints each value in the map.(doseq [x [1 2 3 4 5]] (println x))
: Does the same thing asrun!
in this context.
- Immutability: Clojure’s data structures are immutable. All “changes” to data are creating new data.
(map inc a-vector)
: Returns a new vector with each element incremented, buta-vector
remains unchanged.(filter even? a-list)
: Returns a new list with only the even numbers, buta-list
remains unchanged.(reduce str "" a-set)
: Returns a string that concatenates all the elements in the set, buta-set
remains unchanged.(run! #(println (val %)) a-map)
: Prints each value in the map, buta-map
remains unchanged.(doseq [x a-map] (println (val x)))
: Does the same thing asrun!
in this context, anda-map
remains unchanged.
Last updated on