У меня есть @xmlrootelement , но я продолжаю получать это исключение: невозможно маршалировать тип как элемент, потому что отсутствует @XmlRootElement

Я пытаюсь создать marshall/unmarshall для класса, в котором нет @xmlrootelement. Кто-нибудь может помочь? Маршаллер, который я использую, относится к типу Jaxb2Marshaller org.springframework.oxm.jaxb Spring. Большинство представленных решений относятся к типу javax.xml.bind, Marshaller и JAXBContext. Нужна помощь в marshalling Java-классе, который не имеет предоставленного @xmlrootelement, сгенерированного из XML с использованием Spring JAXB2MARSHALLER.

Я не могу добавить @xmlrootelement, потому что это внешний .jar.

Я пробовал все эти решения, которые на самом деле обеспечивают способ сортировки/десортировки с использованием javax вместо spring.

невозможно маршалировать тип как элемент, потому что отсутствует аннотация @XmlRootElement для автоматически сгенерированных классов

Невозможно маршалировать тип как элемент 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/зависимости, не редактируемый.

Заранее спасибо.

Это разрешило ваш запрос?

Amith Kumar 01.02.2019 18:48

На самом деле это не разрешило мой запрос, но кажется, что у меня есть класс PartnerRequest, который выглядит следующим образом: @XmlAccessorType(XmlAccessType.FIELD) @XmlType( name = "", propOrder = {"partnerCdId", "clientProductCode"} ) @XmlRootElement( name = "PartnerRequest" ) public class PartnerRequest. У него есть @xmlrootelement, но он все еще жалуется на отсутствие аннотации.

Santosh Raviteja 01.02.2019 18:59

Обновил мою проблему. Дайте мне знать, у меня есть @xmlrootelement в PartnerRequest, но все еще исключение показывает проблему для запроса партнера, не имеющего его

Santosh Raviteja 01.02.2019 19:03

Наличие @XmlRootElement полностью вернуло исходный вопрос. Мое оригинальное решение не работает без @XmlRootElement. Позвольте мне включить обработку одного с @XmlRootElement.

Amith Kumar 01.02.2019 20:02

Я также обновил свой репозиторий git примером для одного с корневым элементом. Если вы все еще видите проблему, посмотрите, как инициализируется Jaxb2Marshaller и какие классы/пакеты связаны в контексте? Также, если у вас нет конфликта в классе PartnerRequest в приложении. Если присутствует корневой элемент, он должен работать без проблем.

Amith Kumar 01.02.2019 20:23

Это разрешило ваш запрос? Удалось ли вам добиться желаемого результата? См. Что мне делать, когда кто-то отвечает на мой вопрос?. Как спрашивающий, у вас есть особая привилегия: вы можете принять ответ, который, по вашему мнению, является лучшим решением вашей проблемы.

Amith Kumar 02.02.2019 17:50

Я смог решить эту проблему, но ваша забота определенно помогла. Спасибо

Santosh Raviteja 04.02.2019 16:01

Рад, что смог помочь. Добро пожаловать в Stack Overflow. Обратите внимание, что предпочтительный способ сказать «спасибо» здесь — проголосовать за хорошие вопросы и полезные ответы (если у вас достаточно репутации для этого) и принять наиболее полезный ответ на любой заданный вами вопрос (что также дает вы немного повышаете свою репутацию)

Amith Kumar 04.02.2019 18:40
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
Как вычислять биты и понимать побитовые операторы в Java - объяснение с примерами
Как вычислять биты и понимать побитовые операторы в Java - объяснение с примерами
В компьютерном программировании биты играют важнейшую роль в представлении и манипулировании данными на двоичном уровне. Побитовые операции...
Поднятие тревоги для долго выполняющихся методов в Spring Boot
Поднятие тревоги для долго выполняющихся методов в Spring Boot
Приходилось ли вам сталкиваться с требованиями, в которых вас могли попросить поднять тревогу или выдать ошибку, когда метод Java занимает больше...
Полный курс Java для разработчиков веб-сайтов и приложений
Полный курс Java для разработчиков веб-сайтов и приложений
Получите сертификат Java Web и Application Developer, используя наш курс.
1
8
3 221
4

Ответы 4

Я исправил подобное исключение, используя JAxBElement:

JAXBContext jaxbContext = JAXBContext.newInstance(Foo.class);
Marshaller jaxbMarshaller = jaxbContext.createMarshaller();

Пожалуйста, попробуйте один раз, если это не работает, пожалуйста, поделитесь образцом XSD для дальнейшего изучения.

Как упоминалось ранее, я пробовал это. JAXBCONTEXT происходит от javax.xml.bind. Нужно что-то из SpringFramework

Santosh Raviteja 31.01.2019 19:00

Маршаллинг и демаршаллинг не могут быть выполнены без синтаксического анализатора, который знает корневой элемент. Поэтому, когда у вас нет @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, как и ожидалось.

Другие вопросы по теме