本篇介紹下JAXB進階使用,名稱空間處理
- 使用package-info.java新增預設名稱空間
在需要新增名稱空間的包下面新增package-info.java檔案,然後新增@XmlSchema註解,這樣整個包序列化時就都會自動加上名稱空間了@XmlSchema(namespace = "http://www.lzrabbit.cn") package cn.lzrabbit; import javax.xml.bind.annotation.XmlSchema;
- 名稱空間字首處理
相信大名鼎鼎的ns2,nsXX讓很多人非常頭疼類似下面這樣的<?xml version="1.0" encoding="UTF-8" standalone="yes"?> <ns2:classA xmlns:ns2="http://www.lzrabbit.cn"> <classAId>11</classAId> <ClassAName>A1</ClassAName> <classB> <ClassBId>22</ClassBId> <ClassBName>B2</ClassBName> </classB> </ns2:classA>
解決方法一(不推薦):
新增package-info.java新增@XmlSchema註解並設定屬性xmlns@XmlSchema( xmlns = { @XmlNs(namespaceURI = "http://www.lzrabbit.cn", prefix = "rabbit"), @XmlNs(namespaceURI = "http://www.cnblogs.com", prefix = "blog")}) package cn.lzrabbit; import javax.xml.bind.annotation.XmlSchema; import javax.xml.bind.annotation.XmlNs;
ClassA如下
package cn.lzrabbit; import javax.xml.bind.annotation.*; @XmlRootElement(namespace="http://www.lzrabbit.cn") @XmlAccessorType(XmlAccessType.FIELD) public class ClassA { private int classAId; @XmlElement(name="ClassAName") private String classAName; private ClassB classB; public int getClassAId() { return classAId; } public void setClassAId(int classAId) { this.classAId = classAId; } public String getClassAName() { return classAName; } public void setClassAName(String classAName) { this.classAName = classAName; } public ClassB getClassB() { return classB; } public void setClassB(ClassB classB) { this.classB = classB; } }
序列化結果如下,可以看到已經按照我們所預期的修改了名稱空間字首,這裡要注意下需要自定義字首的實體類新增的@XmlRootElement(namespace="http://www.lzrabbit.cn")註解時指定的namespace必須和package-info.java定義的字首一致,否則還是會生成nsXX這樣的字首
<?xml version="1.0" encoding="UTF-8" standalone="yes"?> <rabbit:classA xmlns:rabbit="http://www.lzrabbit.cn" xmlns:blog="http://www.cnblogs.com"> <classAId>11</classAId> <ClassAName>A1</ClassAName> <classB> <ClassBId>22</ClassBId> <ClassBName>B2</ClassBName> </classB> </rabbit:classA>
注意事項
1.若jdk版本為1.6的需要需要新增jaxb-core-2.2.7.jar和jaxb-impl-2.2.7.jar兩個包的引用,否則即便設定了package-info的XmlSchema註解的xmlns註釋也不能生效,若為jdk 1.7的無需新增
2.使用XmlSchema定義的字首會對整個包生效,無法實現對每個實體類的單獨字首定義,很不靈活,故此不推薦使用此方式
解決方法二(推薦):
同方法一若jdk版本為1.6需要新增jaxb-core-2.2.7.jar和jaxb-impl-2.2.7.jar兩個包的引用,不過方法二不需要新增package-info當然也就不需要定義XmlSchema
思路就是實現NamespacePrefixMapper抽象類,並重寫getPreferredPrefix方法,看到方法名應該都明白了,對就是在序列化的時候重寫獲取名稱空間字首方法,為了簡潔這裡使用類匿名內部類實現的marshaller.setProperty("com.sun.xml.bind.namespacePrefixMapper", new NamespacePrefixMapper() { @Override public String getPreferredPrefix(String namespaceUri, String suggestion, boolean requirePrefix) { if (namespaceUri.equals("http://www.lzrabbit.cn")) return "abc";
return suggestion; } });如上所示,在序列化時判斷namespaceUri也就是我們定義的名稱空間,然後返回我們自定義的字首,其中的suggestion引數就是預設的字首,有興趣的話列印下就會發現suggestion就是ns2之類的字首,把要自定義字首的名稱空間都在這裡判斷下就可以完全控制自定義字首了,相對方法一來說可以實現對每個實體類的名稱空間字首控制,採用方法二後的序列化結果:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?> <abc:classA xmlns:abc="http://www.lzrabbit.cn"> <classAId>11</classAId> <ClassAName>A1</ClassAName> <classB> <ClassBId>22</ClassBId> <ClassBName>B2</ClassBName> </classB> </abc:classA>
採用方法二後的序列化方法
package cn.lzrabbit; import java.io.StringReader; import java.io.StringWriter; import javax.xml.bind.*; import com.sun.xml.bind.marshaller.NamespacePrefixMapper; import com.sun.xml.bind.v2.WellKnownNamespace; public class XmlUtil { public static String toXML(Object obj) { try { JAXBContext context = JAXBContext.newInstance(obj.getClass()); Marshaller marshaller = context.createMarshaller(); marshaller.setProperty(Marshaller.JAXB_ENCODING, "UTF-8");// //編碼格式 marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);// 是否格式化生成的xml串 marshaller.setProperty(Marshaller.JAXB_FRAGMENT, false);// 是否省略xm頭宣告資訊 marshaller.setProperty("com.sun.xml.bind.namespacePrefixMapper", new NamespacePrefixMapper() { @Override public String getPreferredPrefix(String namespaceUri, String suggestion, boolean requirePrefix) { if (namespaceUri.equals("http://www.lzrabbit.cn")) return "abc"; if (namespaceUri.contains("http://www.cnblogs.com")) return "blog"; return suggestion; } }); StringWriter writer = new StringWriter(); marshaller.marshal(obj, writer); return writer.toString(); } catch (Exception e) { throw new RuntimeException(e); } } @SuppressWarnings("unchecked") public static <T> T fromXML(String xml, Class<T> valueType) { try { JAXBContext context = JAXBContext.newInstance(valueType); Unmarshaller unmarshaller = context.createUnmarshaller(); return (T) unmarshaller.unmarshal(new StringReader(xml)); } catch (Exception e) { throw new RuntimeException(e.getMessage()); } } }
現在我們基本解決了jaxb序列化xml的名稱空間及字首問題,但還是有很多問題,比如序列化和反序列化時如何忽略名稱空間,如何使用@XmlRootElement控制每個實體類的預設名稱空間也就是消除名稱空間字首
下一篇繼續深入,Java XML操作之JAXB玩轉名稱空間
最後給下jaxb-core-2.2.7.jar和jaxb-impl-2.2.7.jar兩個包的maven引用
<dependency> <groupId>com.sun.xml.bind</groupId> <artifactId>jaxb-core</artifactId> <version>2.2.7</version> </dependency> <dependency> <groupId>com.sun.xml.bind</groupId> <artifactId>jaxb-impl</artifactId> <version>2.2.7</version> </dependency>
也可以自行去官網下載 https://jaxb.java.net/