Kev*_*vin 6 java rest clojure jaxb jackson
假设我在java中有一个REST API,它支持JSON或XML的响应.响应包含相同的数据,但表单不相同.例如,在json我可能有:
{
"persons": [
{
"name":"Bob",
"age":24,
"hometown":"New York"
}
]
}
Run Code Online (Sandbox Code Playgroud)
而在XML中,它看起来像这样:
<persons>
<person name="bob" age="24">
<hometown>New York</hometown>
</person>
</persons>
Run Code Online (Sandbox Code Playgroud)
也就是说,某些值是人的属性,而其他值是子元素.在Java中,使用JAXB和Jackson,很容易通过模型对象上的注释隐藏这些差异,例如:
public class Person {
@XmlAttribute
String name;
@XmlAttribute
Integer age;
@XmlElement
String hometown;
}
Run Code Online (Sandbox Code Playgroud)
JAXB读取注释,Jackson使用字段名称来确定要做什么.因此,使用单一模型,可以轻松支持多种输出格式.
所以我的问题是,如何在clojure中做同样的事情.我知道有clj-json可以很容易地将clojure地图和矢量转换为json(如果我没弄错的话,使用jackson).我知道clojure.xml.emit和clojure.contrib.xml.prxml都可以将地图和矢量反序列化为XML.但除非我弄错了,否则我认为这两者不会很好地协同工作.
因为prxml要求xml节点表示为向量,而xml属性要表示为映射,这与clj-json的工作根本不同,其中向量表示数组,而映射表示对象.并且clojure.core.emit期望表单中的地图与{:tag :person :attrs {:name "Bob" :age 24} :content ...}clj-json想要的完全不同.
我唯一能想到的是在我的代码中格式化prxml的数据结构,然后编写一个函数,当响应类型为JSON时,该函数将数据结构转换为clj-json想要的内容.但这似乎有点蹩脚.我更喜欢是否有一对JSON和XML库以JAXB和Jackson的方式兼容.
想法?
很大程度上取决于您选择如何在代码中表示模型.
我们假设您使用记录.这是一个人为的示例,说明如何"注释"记录并为XML和JSON提供序列化程序.
;; Depends on cheshire and data.xml
(ns user
(:require [cheshire.core :as json]
[clojure.data.xml :as xml]))
(defrecord Person [name age hometown])
(defrecord Animal [name sound])
(def xml-attrs {Person [:name :age]
Animal [:name]})
(defn record->xml-data [rec]
(let [tag (-> rec class .getSimpleName .toLowerCase keyword)
attrs (select-keys rec (xml-attrs (class rec)))
content (for [[k v] rec
:when (not (contains? attrs k))]
(xml/element k nil (str v)))]
(apply xml/element tag attrs content)))
(defn record->xml [rec]
(xml/emit-str (record->xml-data rec)))
(defn record->json [rec]
(json/generate-string rec))
Run Code Online (Sandbox Code Playgroud)
用法:
> (def bob (Person. "Bob" 24 "New York"))
#'user/bob
> (println (record->xml bob))
<?xml version="1.0" encoding="UTF-8"?><person age="24" name="Bob"><hometown>New York</hometown></person>
nil
> (println (record->json bob))
{"name":"Bob","age":24,"hometown":"New York"}
nil
> (println (record->xml (Animal. "Fido" "Bark!")))
<?xml version="1.0" encoding="UTF-8"?><animal name="Fido"><sound>Bark!</sound></animal>
nil
Run Code Online (Sandbox Code Playgroud)
可以创建一个宏来在单个语句中定义记录及其XML属性.例如,
(defrecord-xml Person [^:xml-attr name ^:xml-attr age hometown])
Run Code Online (Sandbox Code Playgroud)