将函数列表传递给clojure中的'map'会给出nil

时间:2019-03-06 12:12:58

标签: clojure

我正在尝试一些Clojure的例子。

(def sum #(reduce + %))

(def avg #(/ (sum %) (count %)))

(defn stats
  [numbers]
  (map #(% numbers) '(sum, avg)) ;;works when it is [sum avg]
  )

当我调用统计功能时

 (stats [1 24  235 34511 0 14])

它返回(nil nil)。但是,如果我按注释中所述更改代码,它将返回预期的输出。

(34785 11595/2)

为什么不能将函数作为列表传递?

1 个答案:

答案 0 :(得分:7)

您可以将collection参数中的函数传递给map,但是示例中的'前缀是 quoting 列表,因此内容为符号 sumavg代替

'(sum avg)     ;; quoted list, contents are symbols
'[sum avg]     ;; quoted vector, contents are symbols
(list sum avg) ;; list of the functions, using `list` fn to create a list
[sum avg]      ;; vector of the functions

'quote的简写。

未加引号的列表文字被特殊对待。 Clojure将未加引号的列表文字解释为调用,列表中的第一个元素指的是被调用的内容。例如,这将调用sum函数,并将avg函数作为第一个参数传递(这将不起作用):

(sum avg)

通过map对已引用和未引用的列表使用type函数,我们可以看到列表元素类型的不同:

user=> (map type '(conj assoc))
(clojure.lang.Symbol clojure.lang.Symbol)          ;; symbols
user=> (map type (list conj assoc))
(clojure.core$conj__5112 clojure.core$assoc__5138) ;; fn values

这是关于引用的另一种广泛答案:Using quote in Clojure