如何在clojure宏中绑定var的名称和值?

时间:2018-09-11 07:36:08

标签: variables clojure macros lisp

假设我有一些(超过20个)变量,我想将它们保存到文件中。我不想重复20次相同的代码。 我写了一个宏,但它给了我一个错误。

我的测试用例:

;-----------------------------------------------
(defn processor [ some-parameters ]

  (let [
  ;after some operation ,got these data:
        date-str ["JN01","JN02","JN03","JN04"];length 8760
        date-temperature (map #(str %2 "," %1) [3.3,4.4,5.5,6.6] date-str) ; all vector's length are 8760
        date-ws (map #(str %2 "," %1) [0.2,0.1,0.3,0.4] date-str)          ; 
        ;... many variables such like date-relative-humidity,date-pressure, name starts with "date-",
        ; all same size
        ]
    ;(doseq [e date-temperature]
    ;  (println e))
    (spit "output-variable_a.TXT"
          (with-out-str
            (doseq [e date-temperature]
              (println e))))
    ;same 'spit' part will repeat many times
    ))

(processor 123)
; I NEED to output other variables(ws, wd, relative-humidity, ...)
; Output example:
;JN01,3.3
;JN02,4.4
;JN03,5.5
;JN04,6.6
;-----------------------------------------------

我想要的是一个宏/函数,我可以这样使用:

(write-to-text temperature,ws,wd,pressure,theta-in-k,mixradio)

,此宏/功能将完成工作。 我不知道如何编写这样的宏/函数。

我的宏贴在这里,但是不起作用:

(defmacro write-array [& rest-variables ]
  `(doseq [ vname# '~rest-variables ]
     ;(println vname# vvalue#)
     (println "the vname# is" (symbol vname#))
     (println "resolve:" (resolve (symbol (str vname# "-lines"))))
     (println "resolve2:" (resolve (symbol (str "ws-lines"))))
     (let [ vvalue# 5] ;(var-get (resolve (symbol vname#)))]
       ;----------NOTE:  commented out cause '(symbol vname#)' won't work.
       ;1(spit (str "OUT-" vname# ".TXT" )
       ;1      (with-out-str
       ;1        (doseq [ l (var-get (resolve (symbol (str vname# "-lines"))))]
       ;1          (println l))))

       (println vname# vvalue#))))

我发现问题是(symbol vname#)的一部分,此方法仅适用于 GLOBAL 变量,不能绑定到中的 LET 形式,(symbol vname#)返回nil。

3 个答案:

答案 0 :(得分:2)

您似乎想从let内部使用绑定名称及其值编写一个分隔值的文件。宏会在编译期间转换代码,因此它们无法知道您传递的符号所绑定的运行时。您可以使用宏来发出将在运行时评估的代码:

(defmacro to-rows [& args]
  (let [names (mapv name args)]
    `(cons ~names (map vector ~@args))))    

(defn get-stuff []
  (let [nums [1 2 3]
        chars [\a \b \c]
        bools [true false nil]]
    (to-rows nums chars bools)))

(get-stuff)
=> (["nums" "chars" "bools"]
    [1 \a true]
    [2 \b false]
    [3 \c nil])

或者,您可以每行生成一个哈希映射:

(defmacro to-rows [& args]
  (let [names (mapv name args)]
    `(map (fn [& vs#] (zipmap ~names vs#)) ~@args)))

=> ({"nums" 1, "chars" \a, "bools" true}
    {"nums" 2, "chars" \b, "bools" false}
    {"nums" 3, "chars" \c, "bools" nil})

然后,您需要使用data.csv或类似代码将其写到文件中。

要查看to-rows的扩展内容,可以使用macroexpand。这是在编译时生成的 代码,将在运行时进行评估。它完成了在编译时获取符号 names 的工作,但是发出的代码将在运行时对其绑定的 values 起作用。

(macroexpand '(to-rows x y z))
=> (clojure.core/cons ["x" "y" "z"] (clojure.core/map clojure.core/vector x y z))

顺便说一句,我假设您没有在let绑定中键入数千个文字值。我认为这可以回答所问的问题,但是可能有比这更直接的方法。

答案 1 :(得分:0)

我认为您正在寻找函数name。演示:

user=> (defmacro write-columns [& columns]
         (let [names (map name columns)]
           `(str ~@names)))
#'user/write-columns
user=> (write-columns a b c)
"abc"

答案 2 :(得分:0)

您可以先将变量名及其值捕获到映射中:

(defmacro name-map
  [& xs]
  (let [args-list# (cons 'list (map (juxt (comp keyword str) identity) xs))]
    `(into {} ~args-list#)))

如果您将var名称传递给宏,

(let [aa 11
      bb 22
      cc 33]
   (name-map aa bb cc))

它为您提供了一张地图,您可以将其用于任何进一步的处理:

=> {:aa 11, :bb 22, :cc 33}
(def result *1)
(run! 
  (fn [[k v]] (println (str "spit file_" (name k) " value: " v)))
  result)
=>
spit file_aa value: 11
spit file_bb value: 22
spit file_cc value: 33

编辑:只是注意到它类似于泰勒的宏。区别在于,该方法也适用于原始类型,而泰勒(Taylor)则适用于原始数据(将变量解析为集合)。