我在Scala中使用JAXB,编组代码如下所示:
def marshalToXml(): String = {
val context = JAXBContext.newInstance(this.getClass())
val writer = new StringWriter
context.createMarshaller.marshal(this, writer)
writer.toString()
}
然后对于我的可为null的元素,我将@XmlElement(nillable = true)
按照JAXB编组使用带有null字段的注释。这使我的XML输出如下所示:
<name>Alex Dean</name>
<customerReference xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:nil="true"/>
<quantity>1</quantity>
<createdAt>2011-05-14T00:00:00+03:00</createdAt>
这是一个很好的开始,但是我真的想为这些领域编组的是:
<name>Alex Dean</name>
<customerReference nil="true"/>
<quantity type="integer">1</quantity>
<createdAt type="datetime">2011-05-14T00:00:00+03:00</createdAt>
换句话说,我想删除名称空间属性和前缀,并为除字符串之外的所有属性添加显式XML数据类型属性。这可能很简单,但是我似乎无法在JAXB文档中找到方法。
任何帮助表示感谢!
您可以将JAXB与StAX解析器一起使用,并执行以下操作:
顾客
域模型中的每个属性都将映射为@XmlElement(nillable=true, type=Object.class)
。设置type=Object.class
将强制将xsi:type
属性写出。
package forum8198945;
import java.util.Date;
import javax.xml.bind.annotation.*;
@XmlRootElement
public class Customer {
private String name;
private Customer customerReference;
private Integer quantity;
private Date createdAt;
@XmlElement(nillable=true, type=Object.class)
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@XmlElement(nillable=true, type=Object.class)
public Customer getCustomerReference() {
return customerReference;
}
public void setCustomerReference(Customer customerReference) {
this.customerReference = customerReference;
}
@XmlElement(nillable=true, type=Object.class)
public Integer getQuantity() {
return quantity;
}
public void setQuantity(Integer quantity) {
this.quantity = quantity;
}
@XmlElement(nillable=true, type=Object.class)
public Date getCreatedAt() {
return createdAt;
}
public void setCreatedAt(Date createdAt) {
this.createdAt = createdAt;
}
}
XMLStreamWriterWrapper
我们将为创建一个包装器,以XMLStreamWriter
剥离所有我们不想写入XML的信息。
package forum8198945;
import javax.xml.namespace.NamespaceContext;
import javax.xml.stream.XMLStreamException;
import javax.xml.stream.XMLStreamWriter;
public class XMLStreamWriterWrapper implements XMLStreamWriter {
private XMLStreamWriter xmlStreamWriter;
public XMLStreamWriterWrapper(XMLStreamWriter xmlStreamWriter) {
this.xmlStreamWriter = xmlStreamWriter;
}
public void writeStartElement(String localName) throws XMLStreamException {
xmlStreamWriter.writeStartElement(localName);
}
public void writeStartElement(String namespaceURI, String localName) throws XMLStreamException {
xmlStreamWriter.writeStartElement(namespaceURI, localName);
}
public void writeStartElement(String prefix, String localName, String namespaceURI) throws XMLStreamException {
xmlStreamWriter.writeStartElement(prefix, localName, namespaceURI);
}
public void writeEmptyElement(String namespaceURI, String localName) throws XMLStreamException {
xmlStreamWriter.writeEmptyElement(namespaceURI, localName);
}
public void writeEmptyElement(String prefix, String localName, String namespaceURI) throws XMLStreamException {
xmlStreamWriter.writeEmptyElement(prefix, localName, namespaceURI);
}
public void writeEmptyElement(String localName) throws XMLStreamException {
xmlStreamWriter.writeEmptyElement(localName);
}
public void writeEndElement() throws XMLStreamException {
xmlStreamWriter.writeEndElement();
}
public void writeEndDocument() throws XMLStreamException {
xmlStreamWriter.writeEndDocument();
}
public void close() throws XMLStreamException {
xmlStreamWriter.close();
}
public void flush() throws XMLStreamException {
xmlStreamWriter.flush();
}
public void writeAttribute(String localName, String value) throws XMLStreamException {
xmlStreamWriter.writeAttribute(localName, value);
}
public void writeAttribute(String prefix, String namespaceURI, String localName, String value) throws XMLStreamException {
if("http://www.w3.org/2001/XMLSchema-instance".equals(namespaceURI)) {
int colonIndex = value.indexOf(':');
if(colonIndex > -1) {
value = value.substring(colonIndex + 1);
}
xmlStreamWriter.writeAttribute(localName, value);
} else {
xmlStreamWriter.writeAttribute(prefix, namespaceURI, localName, value);
}
}
public void writeAttribute(String namespaceURI, String localName, String value) throws XMLStreamException {
if("http://www.w3.org/2001/XMLSchema-instance".equals(namespaceURI)) {
int colonIndex = value.indexOf(':');
if(colonIndex > -1) {
value = value.substring(colonIndex + 1);
}
xmlStreamWriter.writeAttribute(localName, value);
} else {
xmlStreamWriter.writeAttribute(namespaceURI, localName, value);
}
}
public void writeNamespace(String prefix, String namespaceURI) throws XMLStreamException {
if(!"http://www.w3.org/2001/XMLSchema-instance".equals(namespaceURI) && !"http://www.w3.org/2001/XMLSchema".equals(namespaceURI)) {
xmlStreamWriter.writeNamespace(prefix, namespaceURI);
}
}
public void writeDefaultNamespace(String namespaceURI) throws XMLStreamException {
if(!"http://www.w3.org/2001/XMLSchema-instance".equals(namespaceURI)) {
xmlStreamWriter.writeDefaultNamespace(namespaceURI);
}
}
public void writeComment(String data) throws XMLStreamException {
// TODO Auto-generated method stub
}
public void writeProcessingInstruction(String target) throws XMLStreamException {
// TODO Auto-generated method stub
}
public void writeProcessingInstruction(String target, String data) throws XMLStreamException {
// TODO Auto-generated method stub
}
public void writeCData(String data) throws XMLStreamException {
// TODO Auto-generated method stub
}
public void writeDTD(String dtd) throws XMLStreamException {
// TODO Auto-generated method stub
}
public void writeEntityRef(String name) throws XMLStreamException {
// TODO Auto-generated method stub
}
public void writeStartDocument() throws XMLStreamException {
xmlStreamWriter.writeStartDocument();
}
public void writeStartDocument(String version) throws XMLStreamException {
xmlStreamWriter.writeStartDocument(version);
}
public void writeStartDocument(String encoding, String version) throws XMLStreamException {
xmlStreamWriter.writeStartDocument(encoding, version);
}
public void writeCharacters(String text) throws XMLStreamException {
xmlStreamWriter.writeCharacters(text);
}
public void writeCharacters(char[] text, int start, int len) throws XMLStreamException {
xmlStreamWriter.writeCharacters(text, start, len);
}
public String getPrefix(String uri) throws XMLStreamException {
return xmlStreamWriter.getPrefix(uri);
}
public void setPrefix(String prefix, String uri) throws XMLStreamException {
xmlStreamWriter.setPrefix(prefix, uri);
}
public void setDefaultNamespace(String uri) throws XMLStreamException {
xmlStreamWriter.setDefaultNamespace(uri);
}
public void setNamespaceContext(NamespaceContext context) throws XMLStreamException {
xmlStreamWriter.setNamespaceContext(context);
}
public NamespaceContext getNamespaceContext() {
return xmlStreamWriter.getNamespaceContext();
}
public Object getProperty(String name) throws IllegalArgumentException {
return xmlStreamWriter.getProperty(name);
}
}
XMLStreamReaderWrapper
我们需要为此创建一个包装器,XMLStreamReader
以添加我们在上剥离的所有内容XMLStreamWriter
。XMLStreamReader
因为我们可以扩展,所以这样做更容易StreamReaderDelegate
。
package forum8198945;
import javax.xml.stream.XMLStreamReader;
import javax.xml.stream.util.StreamReaderDelegate;
public class XMLStreamReaderWrapper extends StreamReaderDelegate {
public XMLStreamReaderWrapper(XMLStreamReader xmlStreamReader) {
super(xmlStreamReader);
}
@Override
public String getAttributeNamespace(int index) {
String attributeName = getAttributeLocalName(index);
if("type".equals(attributeName) || "nil".equals(attributeName)) {
return "http://www.w3.org/2001/XMLSchema-instance";
}
return super.getAttributeNamespace(index);
}
}
演示版
以下内容演示了所有内容如何结合在一起:
package forum8198945;
import java.io.StringReader;
import java.io.StringWriter;
import java.util.Date;
import javax.xml.bind.JAXBContext;
import javax.xml.bind.Marshaller;
import javax.xml.stream.XMLInputFactory;
import javax.xml.stream.XMLOutputFactory;
import javax.xml.stream.XMLStreamWriter;
public class Demo {
public static void main(String[] args) throws Exception {
JAXBContext jc = JAXBContext.newInstance(Customer.class);
Customer customer = new Customer();
customer.setName("Alex Dean");
customer.setCustomerReference(null);
customer.setQuantity(1);
customer.setCreatedAt(new Date());
StringWriter stringWriter = new StringWriter();
XMLOutputFactory xof = XMLOutputFactory.newFactory();
XMLStreamWriter xsw = xof.createXMLStreamWriter(stringWriter);
xsw = new XMLStreamWriterWrapper(xsw);
Marshaller marshaller = jc.createMarshaller();
marshaller.marshal(customer, xsw);
String xml = stringWriter.toString();
System.out.println(xml);
XMLInputFactory xif = XMLInputFactory.newFactory();
xif.createXMLStreamReader(new StringReader(xml));
printValue(customer.getName());
printValue(customer.getCustomerReference());
printValue(customer.getQuantity());
printValue(customer.getCreatedAt());
}
private static void printValue(Object value) {
System.out.print(value);
System.out.print(" ");
if(null != value) {
System.out.print(value.getClass());
}
System.out.println();
}
}
输出量
<?xml version="1.0"?><customer><createdAt type="dateTime">2011-11-25T13:36:49.095</createdAt><customerReference nil="true"></customerReference><name type="string">Alex Dean</name><quantity type="int">1</quantity></customer>
Alex Dean class java.lang.String
null
1 class java.lang.Integer
Fri Nov 25 13:36:49 EST 2011 class java.util.Date
首先,我使用Camel 2.15版(在Fuse 6.2.1中)创建了一些路由。 在我的过程中,我试图从使用cxf xjc maven插件生成的pojo中创建一个XML(cxf xjc从xsd的某个地方读取了一些xsd,然后生成了带有jaxb注释的pojo)。 pojos是TempProject和TempProjects。 我可以使用以下代码生成xml: 我打电话 在我实现从pojo到xml的编组的
供应商提供的XML如下: 请注意,没有声明,供应商也没有提供模式。这不能更改,供应商将来会继续这样发布XML。 为了生成JAXB绑定,我创建了如下模式: 请注意,我已经声明了一个或多或少有意义的命名空间(“http://acme.com/schema”),以便它可以用于元素引用等。XJC 生成以下: 然后,我尝试解组XML文档: 我得到的例外是: 显然,这是因为XML元素属于一个空的名称空间,而J
我的代码如下: 我怎样才能让它工作?如果我的xml标记有一些相关元素,如上面的“xmlns”,则JAXB解组器会抛出异常。 这是我第一次这么做,所以我对JAXB、解编等都一无所知。我在苦苦挣扎任何帮助都是好的。 编辑StackTrace:
我的数据有一个ATOM-XML表示,通过Spring MVC web服务返回。我使用JAXB进行序列化,我有许多名称空间,但我希望默认名称空间设置为Atom,不带前缀。下面是我到目前为止在中的内容,但是atom前缀被设置为NS3。 另外,我注意到chrome中显示的名称空间,而Firefox中没有。
命名空间是设计用来扑捉框架最常见用途和提供一个简化和简介的语法用来在应用程序中打开他们。设计是基于框架中的大规模依赖,并且可以划分为以下几个方面: Web/HTTP 安全- 最复杂的部分,设置过滤器和应用框架验证机制的相关服务bean,渲染登录和错误页面等等。 业务对象(方法)安全 - 业务层安全选项. AuthenticationManager - 处理来自框架其他部分的认证请求 AccessD
我试图从通过JAXB生成的类序列化XML。 班级: 文件“package-info.java”: 编组器(简化,无需错误处理): 此代码生成: 我期待这样的事情: 我有一个类似的代码并且工作正常,但是我不明白为什么这个代码不显示命名空间。有线索吗?谢谢!