使用clojure.data.xml从Clojure中发出XML。

时间:2012-05-15 06:07:09

标签: clojure xml-serialization lisp

我需要将Clojure数据结构序列化为XML,但我对如何发布集合感到磕磕绊绊。为了使这更具体,假设我有一个Clojure地图如下:

    (def parking-lot {:parking-lot #{
                         {:car {:color "red", :make "nissan", :year 2003}}
                         {:car {:color "blue", :make "toyota", :year 2001}}
                         {:car {:color "black", :make "honda", :year 2010}}}})

这只是一个带有一个元素的地图,其值是一组“car”项。现在,让我们假设我想序列化这个映射以生成以下XML:

    <? xml version="1.0" ?>
    <parking-lot>
      <car color="red" make="nissan" year="2003" />
      <car color="blue" make="toyota" year="2001" />
      <car color="black" make="black" year="2010" />
    </parking-lot>

在网上搜索有关如何在Clojure中最佳解析/发出XML的文档,让我进入了clojure.data.xml库,这就是我正在使用的。

以下是如何使用clojure.data.xml发出一些XML的简单示例:

    REPL> (use 'clojure.data.xml)

        => nil
    REPL> (emit-str (element :parking-lot {}))

        => "<?xml version=\"1.0\" encoding=\"UTF-8\"?>
            <parking-lot></parking-lot>"

一个稍微复杂的例子:

    REPL> (emit-str (element :parking-lot {}
              (element :car {:color "yellow" :make "ford" :year "2000"})
              (element :car {:color "purple" :make "chevrolet" :year "1977"})))

        => "<?xml version=\"1.0\" encoding=\"UTF-8\"?>
            <parking-lot>
                <car color=\"yellow\" year=\"2000\" make=\"ford\"></car>
                <car color=\"purple\" year=\"1977\" make=\"chevrolet\"></car>
            </parking-lot>"

在这里,您可以看到'element'函数的第三个参数确实可以是对'element'的可变调用次数。如果不深入研究'元素'函数的文档,我想我会假设'元素'会被重载,也会将一系列元素项作为其第三个参数。因此,在没有查看文档的情况下,我只是继续前进并掀起了一小段代码,这些代码就是这样:

    REPL> (emit-str (element :parking-lot {}
              (map (fn [car] (let [carattrs (:car car)]
                       (element :car {:color (:color carattrs),
                                      :make (:make carattrs),
                                      :year (:year carattrs)}))) 
                   (:parking-lot parking-lot))))

我希望这会奏效,但唉不行。问题是'element'函数的第3个参数不能是序列本身。

所以,在这一点上,我有点难过该做什么。我对Lisp和Clojure来说还是比较陌生的,所以如果我的下一个想法是愚蠢的话,请继续关注我。所以我的想法是,这是一个宏的用例吗?即,我应该创建一个宏,它将汽车的一组(或任何碰撞)作为其输入,并将其项目作为单独的s表达式输出?例如,我在脑海中想到一个行为如此的宏:

    REPL> (defmacro flatten-as-elements [coll-of-cars] ...)
    REPL> (flatten-as-elements #{ {:color "blue" :model "honda" year "2000"}
                                  {:color "yellow" :model "ford" year "2011"}})
        => (element :car {:color "blue" :model "honda" year "2000"})
           (element :car {:color "yellow" :model "ford" year "2011"})

至少在我看来,这样一个宏的输出将作为'element'函数的第三个参数,并且会产生我想要的目标。当然,我担心的是有一些完全明显的解决方案,我忽视了,并且在这里使用宏是错误的。非常感谢SO社区的任何帮助!提前谢谢!

-Paul

2 个答案:

答案 0 :(得分:6)

您所要求的转换通常是不可能的,因为您有关于什么应该是属性以及什么是子元素的“隐藏”信息。您可以编写此自定义flatten-as-elements函数或其他任何函数,然后如果在整个过程中插入一堆特定于域的知识,您将能够序列化。

但实际上,你应该在clojure中使用XML的两种流行“表示”之一:将数据写为

{:tag :parking-lot
 :content [{:tag :car :attrs {:color "red", :make "nissan", :year 2003}}
           {:tag :car :attrs {:color "blue", :make "toyota", :year 2001}}
           {:tag :car :attrs {:color "black", :make "honda", :year 2010}}]}

我认为data.xml可以相当简单地为您发出,或者使用更简洁的打嗝样式,带有多个元素的标签和列表的向量:

[:parking-lot (list [:car {:color "red", :make "nissan", :year 2003}]
                    [:car {:color "blue", :make "toyota", :year 2001}]
                    [:car {:color "black", :make "honda", :year 2010}])]

您可以使用(emit (sexp-as-element [...]))进行打印。

答案 1 :(得分:5)

您可以使用apply将参数拼接到函数调用中:

(emit-str
 (apply element :parking-lot {}
        (map (fn [car] (let [carattrs (:car car)]
                         (element :car {:color (:color carattrs),
                                        :make (:make carattrs),
                                        :year (:year carattrs)}))) 
                           (:parking-lot parking-lot))))