clojure.core/str makes sense
While reading excellent Programming Clojure by Stuart Halloway, instead just reading I started looking into how things are done by Clojure itself.Yesterday it took me some time before I understood how things are done by clojure.core/str. This post is documentation of my findings.
Following modification of said function was very helpful in understanding how it works:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(defn mystr | |
"With no args, returns the empty string. With one arg x, returns | |
x.toString(). (str nil) returns the empty string. With more than | |
one arg, returns the concatenation of the str values of the args." | |
{:tag String} | |
([] (println "0:") | |
"") | |
([#^Object x] (println "1: x:" x) | |
(if (nil? x) "" (. x (toString)))) | |
([x & ys] (println "2: x: " x ", ys: " ys) | |
((fn [#^StringBuilder sb more] | |
(if more | |
(do (println "2i: sb: " sb ", more: " more) | |
(recur (. sb (append (mystr (first more)))) (next more))) | |
(do (println "2i: sb: " sb) | |
(mystr sb)))) | |
(new StringBuilder #^String (mystr x)) ys))) |
:tag metadata key is
a symbol naming a class or a Class object that indicates the Java type of the object in the var, or its return value if the object is a fn..
Next are three parameters body pairs.
- ([] "")
- No arguments (arity 0) will result in "" (empty string)
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters. Learn more about bidirectional Unicode charactersuser> (mystr) 0: ""
- ([#^Object x]
(if (nil? x) "" (. x (toString)))) - One argument call (arity 1) will call java.lang.Object.toString method.
#^Object is a Type Hint.
if should not need much explanation.This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters. Learn more about bidirectional Unicode charactersuser> (mystr \a) 1: x: a "a"
- ([x & ys]
((fn [#^StringBuilder sb more]
(if more
(recur (. sb (append (str (first more)))) (next more))
(str sb)))
(new StringBuilder #^String (str x)) ys))) - This is where it starts to be interesting.
First unnamed function is defined that accepts StringBuilder sb and more params.
If more is true, that is not nil or false than call recursively self with new parameters
- sb with appended result of calling str with first element of more,
- more with elements after first.
Next just defined function is getting called with new StringBuilder constructed from x as sb and ys as more.This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters. Learn more about bidirectional Unicode charactersuser> (mystr \a \b) 2: x: a , ys: (b) 1: x: a 2i: sb: #<StringBuilder a> , more: (b) 1: x: b 2i: sb: #<StringBuilder ab> 1: x: #<StringBuilder ab> "ab" This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters. Learn more about bidirectional Unicode charactersuser> (mystr \a \b \c) 2: x: a , ys: (b c) 1: x: a 2i: sb: #<StringBuilder a> , more: (b c) 1: x: b 2i: sb: #<StringBuilder ab> , more: (c) 1: x: c 2i: sb: #<StringBuilder abc> 1: x: #<StringBuilder abc> "abc" - sb with appended result of calling str with first element of more,
It took me some time to realize that ys are turning into more.
So there you have it clojure.core/str explained.