假设我有一些(超过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。
答案 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)则适用于原始数据(将变量解析为集合)。