Download presentation
Presentation is loading. Please wait.
1
Clojure 3 1-Jun-19
2
Clojure errors (NO_SOURCE_FILE:12)
Useless--just means you’re running from the REPL shell java.lang.Exception: EOF while reading (test.clj:139) You have an unclosed parenthesis somewhere Use a good text editor! In jEdit, Control-A Control-I will make the error obvious user=> (map (fn[x] (x * x)) (take 10 (iterate inc 1))) java.lang.ClassCastException: java.lang.Integer cannot be cast to clojure.lang.IFn
3
Sequences In Clojure, almost every collection is a sequence
user=> (cons 5 [1 2 3]) ( ) ; Not [ ] user=> (class '(1 2 3)) clojure.lang.PersistentList user=> (class [1 2 3]) clojure.lang.LazilyPersistentVector user=> (class (cons 5 [1 2 3])) clojure.lang.Cons user=> (class (rest (cons 5 [1 2 3]))) clojure.lang.LazilyPersistentVector$ChunkedSeq Most of the time you don’t care what kind of a sequence it is
4
Persistence and laziness
In Functional Programming, a persistent data structure is one that is itself immutable, but can be modified to create a “new” data structure The original and the new data structure share structure to minimize copying time and wasted storage A lazy data structure is one where parts of it do not exist until they are accessed This allows you to have “infinite” data structures
5
range, take, and drop user=> (range 10) (0 1 2 3 4 5 6 7 8 9)
user=> (take 4 (range 10 20)) ( ) user=> (drop 4 (range 10 20)) ( ) user=> (range 1 10) ( ) user=> (take 20 (range 1 10)) ( ) user=> (drop 20 (range 1 10)) () user=> (take 4 "abcdefg") (\a \b \c \d) user=> (str (take 4 "abcdefg")) user=> (apply str (take 4 "abcdefg")) "abcd"
6
iterate iterate takes a function f and a starting value n, and lazily produces the infinite series (n, f(n), f(f(n)), f(f(f(n))), ...) ; Don’t try this! user=> (iterate inc 1) ( ) ; But this is okay user=> (take 5 (iterate inc 1)) ( ) ; Don’t try this! user=> (drop 5 (iterate inc 1)) ( ) ; But this is okay user=> (take 5 (drop 1000 (iterate inc 1))) ( )
7
Fun with iterate user=> (take 10 (iterate (fn [x] (* 2 x)) 2)) ( ) (defn collatz-1 [n] (cond (= n 1) (even? n) (/ n 2) (odd? n) (inc (* 3 n)) ) ) user=> (take 5 (iterate collatz-1 7)) ( ) user=> (take 25 (iterate collatz-1 7)) ( ) user=> (nth (iterate collatz-1 7) 4) 17
8
Fibonacci numbers user=> (defn next-pair [pair] (list (second pair) (+ (first pair) (second pair)))) #'user/next-pair user=> (next-pair '(1 1)) (1 2) user=> (next-pair '(1 2)) (2 3) user=> (take 10 (iterate next-pair '(0 1))) ((0 1) (1 1) (1 2) (2 3) (3 5) (5 8) (8 13) (13 21) (21 34) (34 55)) user=> (map second (take 10 (iterate next-pair '(0 1)))) ( )
9
Prime numbers user=> (defn divides [d n] (integer? (/ n d))) #'user/divides user=> (divides 4 24) true user=> (divides 5 24) false user=> (defn prime [n] (not-any? (fn [d] (divides d n)) (range 2 (dec n)))) #'user/prime user=> (take 20 (filter prime (iterate inc 2))) ( ) user=> (defn composite [n] (some (fn [d] (divides d n)) (range 2 (dec n)))) #'user/composite user=> (take 20 (filter composite (iterate inc 1))) ( )
10
let and letfn user=> (defn hypotenuse [a b]
(let [a2 (* a a) b2 (* b b)] (Math/sqrt (+ a2 b2)) ) ) #'user/hypotenuse user=> (hypotenuse 3 4) 5.0 user=> (hypotenuse 1 1) user=> (defn hypotenuse [a b] (letfn [(square [n] (* n n))] (Math/sqrt (+ (square a) (square b))))) #'user/hypotenuse user=> (hypotenuse 1 1)
11
Debugging with do user=> (format "%d + %d is %d\n" 3 4 7) "3 + 4 is 7\n" user=> (print (format "%d + %d is %d\n" 3 4 7)) is 7 nil user=> (defn hypotenuse [a b] (letfn [(square [n] (* n n))] (do (println (format "a = %d, b = %d" a b)) (Math/sqrt (+ (square a) (square b))) ) ) ) #'user/hypotenuse user=> (hypotenuse 3 4) a = 3, b =
12
List comprehensions I for macro Usage: (for seq-exprs body-expr) List comprehension. Takes a vector of one or more binding-form/collection-expr pairs, each followed by zero or more modifiers, and yields a lazy sequence of evaluations of expr. Collections are iterated in a nested fashion, rightmost fastest, and nested coll-exprs can refer to bindings created in prior binding-forms. Supported modifiers are: :let [binding-form expr ...], :while test, :when test. user=> (take 12 (for [x (range ) y (range ) :while (< y x)] [x y])) ([1 0] [2 0] [2 1] [3 0] [3 1] [3 2] [4 0] [4 1] [4 2] [4 3] [5 0] [5 1]) Source:
13
List comprehensions II
user=> (map (fn[x] (* x x)) (take 10 (iterate inc 1))) ( ) user=> (for [x (take 10 (iterate inc 1))] (* x x)) ( ) user=> (take 10 (for [x (iterate inc 1)] (* x x))) ( ) user=> (for [x (iterate inc 1) :while (< x 11)] (* x x)) ( ) user=> (for [x (range 1 11)] (* x x)) ( )
14
List comprehensions III
user=> (for [x (range 1 11) :when (even? x)] (* x x)) ( ) user=> (take 10 (for [x (iterate inc 1) :when (even? x)] (* x x))) ( ) ; Don't do this: user=> (for [x (iterate inc 1) :when (even? x)] (* x x)) ( user=> (for [x "abcde" y [1 2]] [x y]) ([\a 1] [\a 2] [\b 1] [\b 2] [\c 1] [\c 2] [\d 1] [\d 2] [\e 1] [\e 2]) user=> (for [word ["the" "quick" "brown" "fox" "jumps"]] (format "Word: %s" word) ) ("Word: the" "Word: quick" "Word: brown" "Word: fox" "Word: jumps")
15
Partial functions (partial f arg1 arg2 arg3 & more) Takes a function f and fewer than the normal arguments to f, and returns a fn that takes a variable number of additional args. When called, the returned function calls f with args + additional args. user=> (def hundred-times (partial * 100)) #'user/hundred-times user=> (hundred-times 5) 500
16
Prime, revisited user=> (defn prime [n] (not-any? (fn [d] (divides d n)) (range 2 (dec n)))) #'user/prime user=> (defn divisible [n d] (integer? (/ n d))) #'user/divisible user=> (divisible 24 3) true user=> (divisible 24 5) false user=> (defn prime [n] (not-any? (partial divisible n) (range 2 (dec n)))) #'user/prime user=> (prime 21) false user=> (prime 19) true
17
Member user=> (defn member [e coll] (cond (empty? coll) false (= e (first coll)) true :else (member e (rest coll)) ) ) #'user/member user=> (member 5 (range 1 10)) true user=> (member 5 (range 10 20)) false
18
Member, revisited user=> (some (partial = 5) (range 1 10)) true
user=> (some (partial = 5) (range 10 20)) nil user=> (defn member [e coll] (some (partial = e) coll)) #'user/member user=> (member 5 (range 1 10)) true user=> (member 5 (range 10 20)) nil
19
Avoiding recursion Java is an object-oriented language
You have a lot of classes available to you You should not write your own Stack class! Clojure is a very recursive language There are a lot of built-in functions that are recursive You should avoid recursion if Clojure will do it for you
20
Zip user=> (zip [1 2 3 4] [:a :b :c]) ((1 :a) (2 :b) (3 :c))
(defn zip [a b] (if (or (empty? a) (empty? b)) () (cons (list (first a) (first b)) (zip (rest a) (rest b))) ) ) user=> (zip [ ] [:a :b :c]) ((1 :a) (2 :b) (3 :c)) user=> (zip '( ) '(:a :b :c)) ((1 :a) (2 :b) (3 :c)) user=> (def zip4 (partial zip [ ])) #'user/zip4 user=> (zip4 "abcde") ((1 \a) (2 \b) (3 \c) (4 \d))
21
Best reference
22
The End
Similar presentations
© 2025 SlidePlayer.com. Inc.
All rights reserved.