когда JAXB unmarshaller.unmarshal возвращает JAXBElement или MySchemaObject?
у меня есть два кода, в двух разных проектах java, делая почти то же самое (unmarshalling вход веб-сервиса в соответствии с xsd-файлом).
но в одном случае я должен написать это: (Input-имя заполнителя) (элемент-вход OMElement)
ClassLoader clInput = input.ObjectFactory.class.getClassLoader();
JAXBContext jc = JAXBContext.newInstance("input", clInput);
Unmarshaller unmarshaller = jc.createUnmarshaller();
Input input = (Input)unmarshaller.unmarshal( element.getXMLStreamReader() );
и в другом lib я должен использовать JAXBElement.getValue (), потому что возвращается JAXBElement, а простой (входной) бросок просто аварийно завершает работу:
Input input = (Input)unmarshaller.unmarshal( element.getXMLStreamReader() ).getValue();
знаете ли вы, что приводит к такая разница ?
5 ответов
если корневой элемент однозначно соответствует классу Java, то будет возвращен экземпляр этого класса, А если нет JAXBElement
будут возвращены.
если вы хотите убедиться, что вы всегда получаете экземпляр объекта домена, вы можете использовать JAXBInstrospector
. Ниже приведен пример.
демо
package forum10243679;
import java.io.StringReader;
import javax.xml.bind.*;
import javax.xml.transform.stream.StreamSource;
public class Demo {
private static final String XML = "<root/>";
public static void main(String[] args) throws Exception {
JAXBContext jc = JAXBContext.newInstance(Root.class);
Unmarshaller unmarshaller = jc.createUnmarshaller();
JAXBIntrospector jaxbIntrospector = jc.createJAXBIntrospector();
Object object = unmarshaller.unmarshal(new StringReader(XML));
System.out.println(object.getClass());
System.out.println(jaxbIntrospector.getValue(object).getClass());
Object jaxbElement = unmarshaller.unmarshal(new StreamSource(new StringReader(XML)), Root.class);
System.out.println(jaxbElement.getClass());
System.out.println(jaxbIntrospector.getValue(jaxbElement).getClass());
}
}
выход
class forum10243679.Root
class forum10243679.Root
class javax.xml.bind.JAXBElement
class forum10243679.Root
Это зависит от присутствии XmlRootElement аннотация о классе вашего корневого элемента.
Если вы создаете классы JAXB из XSD, применяются следующие правила:
- если тип корневого элемента является анонимным типом - > аннотация XmlRootElement добавляется к сгенерированному классу
- если тип корневого элемента является типом верхнего уровня - > xmlrootelement аннотация опущена из созданного класс!--9-->
по этой причине я часто выбираю анонимные типы для корневых элементов.
вы можете настроить имя класса этого анонимного типа с файлом настройки. Е. Г. создать привязки.файл xjc выглядит так:
<jxb:bindings version="1.0"
xmlns:jxb="http://java.sun.com/xml/ns/jaxb"
xmlns:xs="http://www.w3.org/2001/XMLSchema">
<jxb:bindings schemaLocation="yourXsd.xsd" node="/xs:schema">
<jxb:bindings node="//xs:element[@name='yourRootElement']">
<jxb:class name="YourRootElementType"/>
</jxb:bindings>
</jxb:bindings>
</jxb:bindings>
вам нужно добавить в свой класс, созданный JAXB proper @XMLRootElement
- он должен иметь пространство имен:
@XmlRootElement(namespace="http://your.namespace.com/", name="yourRootElement")
посмотрите на связанный вопрос (есть много хороших советов):исключение класса Cast при попытке unmarshall xml?
изменение сгенерированных классов java, я не согласен. Не допуская всех возможных форматов xsd, я не согласен.
Thanx для всех ваших объяснений ссылки, это код, который я написал, чтобы позаботиться о обоих случаях, используя интроспекцию аннотаций. Он работает как для вывода, так и для ввода и является (на мой вкус) более общий:
public class JaxbWrapper {
private static boolean isXmlRootElement(Class classT){
Annotation[] annotations = classT.getAnnotations();
for(Annotation annotation : annotations){
if(annotation instanceof XmlRootElement){
return true;
}
}
return false;
}
public static Object unmarshall(Class classObjectFactory, Class classObject, XMLStreamReader xmlStreamReader){
Package pack = classObjectFactory.getPackage();
String strPackageName = pack.getName();
Object returnObject = null;
try {
JAXBContext jc = JAXBContext.newInstance(strPackageName, classObjectFactory.getClassLoader());
Unmarshaller unmarshaller = jc.createUnmarshaller();
returnObject = unmarshaller.unmarshal( xmlStreamReader );
boolean bIsRootedElement = isXmlRootElement(classObject);
if(!bIsRootedElement)
{
JAXBElement jaxbElement = (JAXBElement) returnObject;
returnObject = jaxbElement.getValue();
}
}
catch (JAXBException e) {
/*...*/
}
return returnObject;
}
private static void writeToXml(Class classObjectFactory, Object obj, XMLStreamWriter xmlStreamWriter){
Package pack = classObjectFactory.getPackage();
String strPackageName = pack.getName();
try {
JAXBContext jc = JAXBContext.newInstance(strPackageName, classObjectFactory.getClassLoader());
Marshaller marshaller = jc.createMarshaller();
marshaller.marshal(obj, xmlStreamWriter);
}
catch(JAXBException e) {
/*...*/
}
}
public static String marshall(Class classObjectFactory, Class classObject, Object obj){
Object objectToMarshall = obj;
boolean bIsRootedElement = isXmlRootElement(classObject);
if(!bIsRootedElement)
{
Package pack = classObjectFactory.getPackage();
String strPackageName = pack.getName();
String strClassName = classObject.getName();
QName qName = new QName(strPackageName, strClassName);
JAXBElement jaxbElement = new JAXBElement(qName, classObject, null, obj);
objectToMarshall = jaxbElement;
}
StringWriter sw = new StringWriter();
XMLOutputFactory xmlOutputFactory = XMLOutputFactory.newInstance();
XMLStreamWriter xmlStreamWriter = null;
try {
xmlStreamWriter = xmlOutputFactory.createXMLStreamWriter(sw);
writeToXml(classObjectFactory, objectToMarshall, xmlStreamWriter);
xmlStreamWriter.flush();
xmlStreamWriter.close();
}
catch (XMLStreamException e) {
/*...*/
}
return sw.toString();
}
}
у меня такая же проблема. JAXB не вызывается.unmarshal возвращает JAXBElement<MyObject>
вместо требуемого MyObject
.
Я нашел и удалены @XmlElementDecl
. Проблема решена.