Clojure数据转换map和reduce

推荐:平凡的你我
推荐语:一首歌,一段故事,一份平凡,蕴含着少许的伟大

说到map和reduce,想到的是著名的MapReduce架构,Clojure的map和reduce在使用上有一定的相似性,但使用更加方便,简洁。

map

map接收一个函数和一个collection作为参数,map的结果是对每一个元素应用到这个函数的一个collection1

(println (map #(str %) [:a :b :c :d]))
;;-> (:a :b :c :d)
;;-> 使用匿名函数将关键字转换为字符,map应用到了每一个元素中,即每一个元素都执行了一遍(#(str %) :a)
(println (class (map #(str %) [:a :b :c :d])))
;;-> clojure.lang.LazySeq

如上显示返回的不是一个向量,而是一个惰性序列,所以我们的map可以处理无穷序列:

(println (take 10(map #(str %) (range))))
;;-> (0 1 2 3 4 5 6 7 8 9)
;;-> 输出的元素为字符

注意我们使用take获取到前十个元素,因为不设定个数就会参数无穷序列。

当函数为println时,执行下列代码并不会打印出数据,这就是惰性,不会立马运算出,可以理解为接受参数并打印出来是一个副作用,惰性会使得不会触发这个副作用,可能不太好理解,该部分先记清就行,带副作用的函数和惰性结合需要小心,副作用有可能不被执行。

(def aa (map #(println %) ["aa" "bb" "cc"]))

此时,使用doall可以强制化执行:

(def aa (doall (map #(println %) ["aa" "bb" "cc"])))
;;-> aa
;;-> bb
;;-> cc

map可以接收不止一个collection,如果有多个collection,就会分别取一个元素作为函数的参数:

(defn example [a b]
  (str "a is " a ";b is " b))
(println (map example ["vv" "cc"] ["dd" "gg"]))
;;-> (a is vv;b is dd a is cc;b is gg)

程序的结束根据短的collection长度。

reduce

先看一个经典的例子:

(println (reduce + [1 2 3 4 5 6 7 8]))
;;-> 36
(println (reduce (fn [r x] (+ r (* x x))) [2 3 4]))
;;-> 27

首先将后续的向量中的第一个元素作为初始元素,即2赋值给r,然后再将r+3x3+4x4,结果等于27。

(println (reduce
            (fn [r x]
              (if (nil? x) r (conj r x)))
            []
            [:a nil :b nil :c]))
;;-> [:a :b :c]

同上一个类似,r初始赋值为[],将[:a nil :b nil :c]中的nil过滤掉,输出结果为新的向量。

其它一些函数

complement

返回相反的布尔值

(println ((complement nil?) nil))
;;-> false
filter

真的结果输出

(println (filter #(> 3 %) [1 2 3 4 5]))
;;-> (1 2)
(println (filter (complement nil?) [:a :b :c nil]))
;;-> (:a :b :c)
(println (filter keyword? [:a :b :c 1 2]))
;;-> (:a :b :c)
remove

剔除结果为真的元素

(println (remove nil? [:a nil :b nil :c]))
;;-> (:a :b :c)
for

有点类似foreach,先绑定参数,再依次输出collection的元素:

(println (for [s [:a :b :c :d :e]]
           (str "str = " s)))
;;-> (str = :a str = :b str = :c str = :d str = :e)
;;-> 多参数类似多层循环
(println (for [s1 [:a :b :c :d :e]
              s2 [:xx :zz :vv]]
           (str " s1=" s1 " s2=" s2)))
;;-> ( s1=:a s2=:xx  s1=:a s2=:zz  s1=:a s2=:vv  s1=:b s2=:xx  s1=:b s2=:zz  s1=:b s2=:vv  s1=:c s2=:xx  s1=:c s2=:zz  s1=:c s2=:vv  s1=:d s2=:xx  s1=:d s2=:zz  s1=:d s2=:vv  s1=:e s2=:xx  s1=:e s2=:zz  s1=:e s2=:vv)

在for[]内部还可以使用:let 和:when,:let是用来绑定数据的,:when是用来说明执行条件,为真时下方的语句才会被执行2

(for [x [0 1 2 3 4 5]
      :let [y (* x 3)]
      :when (even? y)]
  y)
;;=> (0 6 12)
(filter even? (range 10))
;;-> (0 2 4 6 8)
;;-> 返回偶数
vec

生成新的向量

(println (vec '(1 2 3)))
;;-> [1 2 3]
into

加入元素到[]向量

(println (into [] '(1 2 3)))
;;-> [1 2 3]
partition

分割一个collection的函数

(println (partition 2 [1 2 3 4 5 6]))
;;-> ((1 2) (3 4) (5 6))
(println (partition 3 [1 2 3 4 5 6 7]))
;;-> ((1 2 3) (4 5 6))
;;-> 使用partition-all不会丢弃元素
(println (partition-all 3 [1 2 3 4 5 6 7]))
;;-> ((1 2 3) (4 5 6) (7))
;;-> partition-by [f] 输入函数,满足函数的地方进行分割
(println (partition-by #(= 3 %) [1 2 3 4 5 6 7]))

推荐

http://clojuredocs.org/clojure.core 是clojure的官方文档,里面有很多的例子,虽然是英文但也大致能够看懂,当遇到困惑时可以直接在里面搜索你想了解的内容,里面的例子也有很大的参考价值。


  1. Living Clojure(中文版):中国电力出版社 ↩︎

  2. clojure core docs:http://clojuredocs.org/clojure.core/for ↩︎

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Clojure 和 Python 是两种不同的编程语言,它们具有不同的设计哲学和特点。以下是对 Clojure 和 Python 进行比较的一些方面: 1. 语法和风格: - Clojure 是一种 Lisp 方言,其语法简洁、统一且拥有强大的宏系统。它使用括号表示表达式,具有强调函数式编程和不可变性的风格。 - Python 的语法更接近自然语言,使用缩进来表示代码块,具有直观和易读的风格。它既支持面向对象编程,也支持函数式编程。 2. 类型系统: - Clojure 是一种动态类型语言,变量的类型可以在运行时改变。它采用了基于值的不可变数据结构,并通过特定的函数操作来实现数据转换和处理。 - Python 也是一种动态类型语言,但它还支持静态类型检查(通过类型注解)。它使用了更传统的可变数据结构,并提供了丰富的内置数据类型和操作。 3. 并发和并行处理: - Clojure 在语言层面上支持并发和并行处理。它提供了一些内置的机制(如原子操作、锁和通道)来处理共享状态和并发访问。 - Python 的并发和并行处理需要借助于第三方库(如 threading、multiprocessing 和 asyncio)来实现。它提供了多线程和多进程编程的支持,以及异步编程模型。 4. 生态系统和库支持: - Python 拥有非常庞大和活跃的生态系统,有丰富的第三方库和工具可供选择,用于各种领域的开发,如科学计算、Web 开发、数据分析等。 - Clojure 的生态系统相对较小,但也有一些优秀的库和工具可供使用。它可以与 Java 生态系统进行互操作,可以利用 Java 的库和工具来扩展功能。 总体来说,Clojure 和 Python 在语法、类型系统和并发处理方面存在差异。Python 是一种通用的、易于学习和使用的语言,适合于快速开发和脚本编写。Clojure 则更适合于函数式编程和并发处理,对于处理大规模数据和构建高性能应用程序具有优势。选择哪种语言取决于具体的需求和个人偏好。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值