Abstraction

在了解sequence之前,我们可以先了解下abstraction,abstraction的概念在很多语言里面都有,譬如Go,interface就是abstraction:

type IA interface {
    DoFunc()
}

type A struct {}

func (a *A) DoFunc() {

}

在上面这个例子中,struct A实现了DoFunc函数,我们就可以认为,A实现了IA。

Sequence Abstraction

Clojure也提供了abstraction的概念,这里我们主要来了解下sequence abstraction。

在Clojure里面,如果这些core sequence function first,rest和cons能够用于某个data structure,我们就可以认为这个data structure实现了sequence abstraction,就能被相关的sequence function进行操作,譬如map,reduce等。

First

first返回collection里面的第一个元素,譬如:

user=> (first [1 2 3])
1
user=> (first '(1 2 3))
1
user=> (first #{1 2 3})
1
user=> (first {:a 1 :b 2})
[:a 1]

Rest

rest返回collection里面,第一个元素后面的sequence,譬如:

user=> (rest [1 2 3])
(2 3)
user=> (rest [1])
()
user=> (rest '(1 2 3))
(2 3)
user=> (rest #{1 2 3})
(3 2)
user=> (rest {:a 1 :b 2})
([:b 2])

Cons

Cons则是将一个元素添加到collection的开头,譬如:

user=> (cons 1 [1 2 3])
(1 1 2 3)
user=> (cons 1 '(1 2 3))
(1 1 2 3)
user=> (cons 1 #{1 2 3})
(1 1 3 2)
user=> (cons 1 {:a 1 :b 2})
(1 [:a 1] [:b 2])
user=> (cons {:c 3} {:a 1 :b 2})
({:c 3} [:a 1] [:b 2])

从上面的例子可以看出,Clojure自身的vector,list等都实现了sequence abstraction,所以他们也能够被一些sequence function调用:

user=> (defn say [name] (str "hello " name))
#'user/say
user=> (map say [1 2])
("hello 1" "hello 2")
user=> (map say '(1 2))
("hello 1" "hello 2")
user=> (map say #{1 2})
("hello 1" "hello 2")
user=> (map say {:a 1 :b 2})
("hello [:a 1]" "hello [:b 2]")
user=> (map #(say (second %)) {:a 1 :b 2})
("hello 1" "hello 2")

Collection Abstraction

跟sequence abstraction类似,Clojure里面的core data structure,譬如vector,list等,都实现了collection abstraction。

Collection abstraction通常是用于处理整个data structure的,譬如:

user=> (count [1 2 3])
3
user=> (empty? [])
true
user=> (every? #(< % 3) [1 2 3])
false
user=> (every? #(< % 4) [1 2 3])
true

into

一个重要的collection function就是into,sequence function通常会返回一个seq,而into会将返回的seq转换成原始的data structure,譬如:

user=> (into [] [1 2 3])
[1 2 3]
user=> (into [] (map inc [1 2 3]))
[2 3 4]