我有一个如下所示的XML文档,并希望对其执行解组
<?xml version="1.0" encoding="UTF-8"?>
<SOAP-ENV:Envelope xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:SOAP-ENC="http://schemas.xmlsoap.org/soap/encoding/"
xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:sh="http://www.w3.org/2001/XMLSchema-instance">
<SOAP-ENV:Body>
<people xmlns="http://ccm.intra.bt.com/manageServiceFault/2006/06"
xmlns:cds="http://capabilities.nat.bt.com/xsd/manageServiceFault/2010/06/Contact/Details"
xmlns:sh="http://wsi.nat.bt.com/2005/06/StandardHeader/"
xsi:schemaLocation="http://ccm.intra.bt.com/manageServiceFault/2006/06 MSF_5.0.xsd">
<cds:address>
<cds:Address>
<cds:postTown>London</cds:postTown>
<cds:postCode>WC2E 7AT</cds:postCode>
<cds:thoroughfareName>Bow Street</cds:thoroughfareName>
<cds:dependentLocality>local123</cds:dependentLocality>
<cds:county>London</cds:county>
<cds:thoroughfareNumber>30-34</cds:thoroughfareNumber>
<cds:subPremise>2nd Floor</cds:subPremise>
<cds:buildingName>BT tower</cds:buildingName>
<cds:buildingNumber>Lot 1234</cds:buildingNumber>
<cds:locality>London</cds:locality>
<cds:premise>Covent Garden Exchange</cds:premise>
<cds:dependentThoroughfare>345</cds:dependentThoroughfare>
<cds:poBox>PO1234</cds:poBox>
</cds:Address>
<cds:Address>
<cds:postTown>New York</cds:postTown>
<cds:postCode>WC2E 7AT</cds:postCode>
<cds:thoroughfareName>Bow Street</cds:thoroughfareName>
<cds:dependentLocality>local123</cds:dependentLocality>
<cds:county>US</cds:county>
<cds:thoroughfareNumber>30-34</cds:thoroughfareNumber>
<cds:subPremise>2nd Floor</cds:subPremise>
<cds:buildingName>BT tower</cds:buildingName>
<cds:buildingNumber>Lot 1234</cds:buildingNumber>
<cds:locality>US</cds:locality>
<cds:premise>Covent Garden Exchange</cds:premise>
<cds:dependentThoroughfare>345</cds:dependentThoroughfare>
<cds:poBox>PO1234</cds:poBox>
</cds:Address>
<cds:country>
<cds:name>UK</cds:name>
</cds:country>
<cds:country>
<cds:name>US</cds:name>
</cds:country>
</cds:address>
</people>
</SOAP-ENV:Body>
</SOAP-ENV:Envelope>
请注意,元素地址包含一个或多个Address和Country元素。地址和地址都相同,只是A差异。以下是我的域名类
People2.java
import javax.xml.bind.annotation.XmlAccessType;
import javax.xml.bind.annotation.XmlAccessorType;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlRootElement;
import javax.xml.bind.annotation.XmlType;
@XmlRootElement(name = "people")
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(namespace = "http://capabilities.nat.bt.com/xsd/manageServiceFault/2010/06/Contact/Details")
public class People2 {
@XmlElement(name = "address")
private Addresses addresses;
public Addresses getAddresses() {
return addresses;
}
public void setAddresses(Addresses addresses) {
this.addresses = addresses;
}
}
Addresses.java
import java.util.ArrayList;
import java.util.List;
import javax.xml.bind.annotation.XmlAccessType;
import javax.xml.bind.annotation.XmlAccessorType;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlElementWrapper;
import javax.xml.bind.annotation.XmlType;
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(namespace = "http://capabilities.nat.bt.com/xsd/manageServiceFault/2010/06/Contact/Details")
public class Addresses {
@XmlElementWrapper(name = "address")
@XmlElement(name = "Address")
private List<Address> address;
@XmlElementWrapper(name = "address")
@XmlElement(name = "country")
private List<Country> country;
public Addresses() {
address = new ArrayList<Address>();
country = new ArrayList<Country>();
}
public List<Address> getAddress() {
return address;
}
public void setAddress(List<Address> address) {
this.address = address;
}
public List<Country> getCountry() {
return country;
}
public void setCountry(List<Country> country) {
this.country = country;
}
}
及以下是我解组XML的主要代码
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.List;
import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBException;
import javax.xml.bind.Marshaller;
import javax.xml.bind.Unmarshaller;
import javax.xml.stream.XMLInputFactory;
import javax.xml.stream.XMLStreamConstants;
import javax.xml.stream.XMLStreamException;
import javax.xml.stream.XMLStreamReader;
public class Demo2 {
public static void main(String[] args) throws FileNotFoundException, XMLStreamException, JAXBException,
IllegalAccessException, IllegalArgumentException, InvocationTargetException, IntrospectionException {
XMLInputFactory xif = XMLInputFactory.newInstance();
XMLStreamReader xsr = xif.createXMLStreamReader(new FileReader("./xml/Testing2.xml"));
JAXBContext jc = JAXBContext.newInstance(People2.class);
Unmarshaller unmarshaller = jc.createUnmarshaller();
xsr.nextTag();
System.out.println("Current XML element: " + xsr.getLocalName());
xsr.nextTag();
System.out.println("Current XML element: " + xsr.getLocalName());
int tag = xsr.nextTag();
if (tag == XMLStreamConstants.START_ELEMENT) {
System.out.println("Current XML element: " + xsr.getLocalName());
People2 people = (People2) unmarshaller.unmarshal(xsr);
for (PropertyDescriptor propertyDescriptor : Introspector.getBeanInfo(People2.class, Object.class)
.getPropertyDescriptors()) {
Method method = propertyDescriptor.getReadMethod();
System.out.println("Class name: " + people.getClass().getSimpleName() + ", Method: " + method.getName()
+ ", Value: " + method.invoke(people));
}
Addresses addresses = people.getAddresses();
for (PropertyDescriptor propertyDescriptor : Introspector.getBeanInfo(Addresses.class, Object.class)
.getPropertyDescriptors()) {
Method method = propertyDescriptor.getReadMethod();
System.out.println("Class name: " + addresses.getClass().getSimpleName() + ", Method: "
+ method.getName() + ", Value: " + method.invoke(addresses));
}
}
}
}
问题是,当在地址中打印方法时,getAddress和getCountry的值返回null,如下所示。似乎绑定没有发生。
Current XML element: Envelope
Current XML element: Body
Current XML element: people
Class name: People2, Method: getAddresses, Value: com.bt.platform.automation.domain.Addresses@433c675d
Class name: Addresses, Method: getAddress, Value: []
Class name: Addresses, Method: getCountry, Value: []
如果我在Addresses.java中删除此@XmlElementWrapper(name = "address")
,它可以正常工作,如下所示
Current XML element: Envelope
Current XML element: Body
Current XML element: people
Class name: People2, Method: getAddresses, Value: com.bt.platform.automation.domain.Addresses@2e817b38
Class name: Addresses, Method: getAddress, Value: [com.bt.platform.automation.domain.Address@c4437c4, com.bt.platform.automation.domain.Address@433c675d]
Class name: Addresses, Method: getCountry, Value: [com.bt.platform.automation.domain.Country@3f91beef, com.bt.platform.automation.domain.Country@1a6c5a9e]
如果我不删除@XmlElementWrapper(name = "address")
,那么有人请帮忙,为什么它没有按预期工作?我已经提到了这篇文章 - http://blog.bdoughan.com/2010/09/jaxb-collection-properties.html,但它似乎不适用于我的代码。
答案 0 :(得分:0)
我认为你误解了@XmlElementWrapper的用途。
为了举例,我们假设你的地址元素只包含地址元素。
然后您可以使用以下代码,并删除Adresses类的必要性:
@XmlRootElement(name = "people")
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(namespace = "http://capabilities.nat.bt.com/xsd/manageServiceFault/2010/06/Contact/Details")
public class People2 {
@XmlElement(name = "Address")
@XmlElementWrapper(name = "address")
private List<Address> addresses;
public List<Address> getAddresses() {
return addresses;
}
public void setAddresses(List<Address> addresses) {
this.addresses = addresses;
}
}
为了这个原因,存在@XmlElementWrapper,因为我们倾向于在给定元素列表周围放置一个包装元素,因此它会强制使用中间类。