在`case`中使用channel

时间:2017-03-06 20:52:21

标签: clojure core.async

(let [a (clojure.core.async/chan)]
  (case a
    a :foo
    :bar))
#=> :bar

我希望:foo在这里。我做错了什么?

另一方面,(condp = chan ...)完成了这项工作。

PS:

基本上我正在尝试做以下事情:

(require '[clojure.core.async :as a])
(let [chan1 (a/chan 10)
      chan2 (a/chan 10)]
  (a/>!! chan1 true)
  (let [[v c] (a/alts!! [chan1 chan2])]
    (case c
      chan1 :chan1
      chan2 :chan2
      :niether)))
#=> :neither

2 个答案:

答案 0 :(得分:4)

The docs for case有答案

  

不评估测试常数。它们必须是编译时   文字,不需要引用。

正确的解决方案是使用cond

(let [chan1 (ca/chan 10)
      chan2 (ca/chan 10)]
    (ca/>!! chan1 true)
    (let [[v c] (ca/alts!! [chan1 chan2])]
      (spyx (cond
              (= c chan1) :chan1
              (= c chan2) :chan2
              :else :neither))))

;=> :chan1

答案 1 :(得分:1)

Case使用未评估的测试常量作为子句的左侧。像chan1这样的普通符号只匹配具有相同名称的符号,而不匹配具有该名称的本地绑定的值; chan1将匹配'chan1