Я пытаюсь создать marshall/unmarshall для класса, в котором нет @xmlrootelement. Кто-нибудь может помочь? Маршаллер, который я использую, относится к типу Jaxb2Marshaller org.springframework.oxm.jaxb Spring. Большинство представленных решений относятся к типу javax.xml.bind, Marshaller и JAXBContext. Нужна помощь в marshalling Java-классе, который не имеет предоставленного @xmlrootelement, сгенерированного из XML с использованием Spring JAXB2MARSHALLER.
Я не могу добавить @xmlrootelement, потому что это внешний .jar.
Я пробовал все эти решения, которые на самом деле обеспечивают способ сортировки/десортировки с использованием javax вместо spring.
Невозможно маршалировать тип как элемент XML, так как отсутствует аннотация @XmlRootElement
Включая это, которое имеет очень аккуратное объяснение, но использует Marshaller вместо JaxB2Marshaller.
https://howtodoinjava.com/jaxb/marshal-без-xmlrootelement/
Класс должен пройти Маршаллинг без @xmlrootelement
Класс PartnerRequest.java с @xmlrootelement
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(
name = "",
propOrder = {"partnerCdId", "clientProductCode"}
)
@XmlRootElement(
name = "PartnerRequest"
)
public class PartnerRequest
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(
name = "Partner",
)
public class Partner{
}
Исключение, которое я получаю, выглядит следующим образом:
[com.sun.istack.internal.SAXException2: unable to marshal type "com.svc.partner.v1.PartnerRequest" as an element because it is missing an @XmlRootElement annotation]
at org.springframework.oxm.jaxb.Jaxb2Marshaller.convertJaxbException(Jaxb2Marshaller.java:933)
at org.springframework.oxm.jaxb.Jaxb2Marshaller.marshal(Jaxb2Marshaller.java:709)
at org.springframework.ws.support.MarshallingUtils.marshal(MarshallingUtils.java:81)
at org.springframework.ws.client.core.WebServiceTemplate$2.doWithMessage(WebServiceTemplate.java:395)
at org.springframework.ws.client.core.WebServiceTemplate.doSendAndReceive(WebServiceTemplate.java:573)
at org.springframework.ws.client.core.WebServiceTemplate.sendAndReceive(WebServiceTemplate.java:539)
at org.springframework.ws.client.core.WebServiceTemplate.marshalSendAndReceive(WebServiceTemplate.java:386)
at org.springframework.ws.client.core.WebServiceTemplate.marshalSendAndReceive(WebServiceTemplate.java:380)
at com.vmd.services.BaseService.marshallSendAndReceive(BaseService.java:53)
at com.vmd.services.PartnerServiceImpl.getPartner(PartnerServiceImpl.java:27)
at com.vmd.services.PartnerServiceImplTest.ShouldGetPartner(PartnerServiceImplTest.java:36)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
at org.springframework.test.context.junit4.statements.RunBeforeTestExecutionCallbacks.evaluate(RunBeforeTestExecutionCallbacks.java:74)
at org.springframework.test.context.junit4.statements.RunAfterTestExecutionCallbacks.evaluate(RunAfterTestExecutionCallbacks.java:84)
at org.springframework.test.context.junit4.statements.RunBeforeTestMethodCallbacks.evaluate(RunBeforeTestMethodCallbacks.java:75)
at org.springframework.test.context.junit4.statements.RunAfterTestMethodCallbacks.evaluate(RunAfterTestMethodCallbacks.java:86)
at org.springframework.test.context.junit4.statements.SpringRepeat.evaluate(SpringRepeat.java:84)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:251)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:97)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
at org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61)
at org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:70)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:190)
at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68)
at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:47)
at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70)
Caused by: javax.xml.bind.MarshalException: null
at com.sun.xml.internal.bind.v2.runtime.MarshallerImpl.write(MarshallerImpl.java:311)
at com.sun.xml.internal.bind.v2.runtime.MarshallerImpl.marshal(MarshallerImpl.java:236)
at org.springframework.oxm.jaxb.Jaxb2Marshaller.marshal(Jaxb2Marshaller.java:705)
... 39 common frames omitted
Caused by: com.sun.istack.internal.SAXException2: unable to marshal type "com.svc.partner.v1.PartnerRequest" as an element because it is missing an @XmlRootElement annotation
at com.sun.xml.internal.bind.v2.runtime.XMLSerializer.reportError(XMLSerializer.java:234)
at com.sun.xml.internal.bind.v2.runtime.ClassBeanInfoImpl.serializeRoot(ClassBeanInfoImpl.java:323)
at com.sun.xml.internal.bind.v2.runtime.XMLSerializer.childAsRoot(XMLSerializer.java:479)
at com.sun.xml.internal.bind.v2.runtime.MarshallerImpl.write(MarshallerImpl.java:308)
... 41 common frames omitted
Кроме того, помните, что нам нужно маршаллировать его, используя Jaxb2Mashaller, а не Marshaller. Это внешний код jar/зависимости, не редактируемый.
Заранее спасибо.
На самом деле это не разрешило мой запрос, но кажется, что у меня есть класс PartnerRequest, который выглядит следующим образом: @XmlAccessorType(XmlAccessType.FIELD) @XmlType( name = "", propOrder = {"partnerCdId", "clientProductCode"} ) @XmlRootElement( name = "PartnerRequest" ) public class PartnerRequest. У него есть @xmlrootelement, но он все еще жалуется на отсутствие аннотации.
Обновил мою проблему. Дайте мне знать, у меня есть @xmlrootelement в PartnerRequest, но все еще исключение показывает проблему для запроса партнера, не имеющего его
Наличие @XmlRootElement полностью вернуло исходный вопрос. Мое оригинальное решение не работает без @XmlRootElement. Позвольте мне включить обработку одного с @XmlRootElement.
Я также обновил свой репозиторий git примером для одного с корневым элементом. Если вы все еще видите проблему, посмотрите, как инициализируется Jaxb2Marshaller и какие классы/пакеты связаны в контексте? Также, если у вас нет конфликта в классе PartnerRequest в приложении. Если присутствует корневой элемент, он должен работать без проблем.
Это разрешило ваш запрос? Удалось ли вам добиться желаемого результата? См. Что мне делать, когда кто-то отвечает на мой вопрос?. Как спрашивающий, у вас есть особая привилегия: вы можете принять ответ, который, по вашему мнению, является лучшим решением вашей проблемы.
Я смог решить эту проблему, но ваша забота определенно помогла. Спасибо
Рад, что смог помочь. Добро пожаловать в Stack Overflow. Обратите внимание, что предпочтительный способ сказать «спасибо» здесь — проголосовать за хорошие вопросы и полезные ответы (если у вас достаточно репутации для этого) и принять наиболее полезный ответ на любой заданный вами вопрос (что также дает вы немного повышаете свою репутацию)




Я исправил подобное исключение, используя JAxBElement:
JAXBContext jaxbContext = JAXBContext.newInstance(Foo.class);
Marshaller jaxbMarshaller = jaxbContext.createMarshaller();
Пожалуйста, попробуйте один раз, если это не работает, пожалуйста, поделитесь образцом XSD для дальнейшего изучения.
Как упоминалось ранее, я пробовал это. JAXBCONTEXT происходит от javax.xml.bind. Нужно что-то из SpringFramework
Маршаллинг и демаршаллинг не могут быть выполнены без синтаксического анализатора, который знает корневой элемент. Поэтому, когда у вас нет @XmlRootElement в классе данных, вы обходите использование JAXBElement, где вы предоставляете свой класс модели, обернутый в него, чтобы маршаллер знал, на какую структуру исходного класса ссылаться при синтаксическом анализе.
Ниже я привел рабочий пример использования JAXBElement с Spring Jaxb2Marshaller.
@SpringBootApplication
public class XmlTransformationNoRoot {
public static void main(String[] args) {
SpringApplication.run(XmlTransformationNoRoot.class, args);
}
}
@Component
class TestTransformation implements CommandLineRunner {
public void run(String[] args) throws JAXBException {
//marshalling
String testXML = marshallXml(new Partner("HelloWorld"));
System.out.println("Marshalled : " + testXML);
//unmarshalling
System.out.println("Unmarshalled : " + unmarshallXml(testXML, Partner.class));
}
public <T> String marshallXml(T object) throws JAXBException {
StringWriter sw = new StringWriter();
Result result = new StreamResult(sw);
jaxb2Marshaller().marshal(new JAXBElement(
new QName(object.getClass().getPackage().getName(), object.getClass().getSimpleName()),
object.getClass(), object), result);
return sw.toString();
}
@SuppressWarnings("unchecked")
public <T> T unmarshallXml(String source, Class<T> clazz) throws JAXBException {
Jaxb2Marshaller marshaller = jaxb2Marshaller();
//this line voids singleton use of Jaxb2Marshaller
marshaller.setMappedClass(clazz);
return (T) marshaller.unmarshal(new StreamSource(new StringReader(source)));
}
public Jaxb2Marshaller jaxb2Marshaller() {
Jaxb2Marshaller marshaller = new Jaxb2Marshaller();
marshaller.setMarshallerProperties(new HashMap() {{
put(javax.xml.bind.Marshaller.JAXB_FORMATTED_OUTPUT, true);
}});
marshaller.setClassesToBeBound(new Class[]{Partner.class});
return marshaller;
}
}
Надеюсь это поможет. Исполняемый исходный код доступен в моем репозитории гитхаб.
Наконец-то мне удалось решить проблему. Проблема была с именем пакета, и я, похоже, импортировал неправильный пакет, для которого он жаловался на то, что partnerRequest отсутствует @xmlrootelement.
Итак, наконец, нам нужно быть осторожными, когда у нас есть два класса с одинаковым именем и при импорте с правильными пакетами. Всем спасибо!
В весенней загрузке 2 я обычно использую
@Bean
MessageConverter getMessageConveter() {
Jaxb2Marshaller jaxb2Marshaller = new Jaxb2Marshaller();
jaxb2Marshaller.setContextPath(ObjectFactory.class.getPackage().getName());
Map<String, Object> prop = new HashMap<>();
prop.put(Marshaller.JAXB_FORMATTED_OUTPUT, Boolean.TRUE);
jaxb2Marshaller.setMarshallerProperties(prop);
jaxb2Marshaller.setCheckForXmlRootElement(false);
return new MarshallingMessageConverter(jaxb2Marshaller);
}
Однако все еще получал ошибку. Так было до тех пор, пока я не обнаружил, что пытаюсь маршалировать сложный тип. Однажды я завернул этот объект в
<xs:element name = "..."
он сгенерировал аннотацию @XMLRootElement, как и ожидалось.
Это разрешило ваш запрос?