Bootstrap Hibernate SessionFactory в методе обработчика Netty/Armeria завершается с ошибкой ClassNotFoundException
У меня есть приложение Java, которое использует Armeria для веб-службы. Когда я создаю свой Hibernate SessionFactory в основном методе, он отлично работает. Но я пытаюсь создать SessionFactory, когда вызывается определенная конечная точка Http. В методе обработчика нельзя создать фабрику сеансаException in thread "Thread-1" org.hibernate.internal.util.config.ConfigurationException: Unable to perform unmarshalling at line number 0 and column 0 in RESOURCE hibernate.cfg.xml. Message: null
Caused by: javax.xml.bind.JAXBException
- with linked exception:
[java.lang.ClassNotFoundException: com/sun/xml/bind/v2/ContextFactory]
at javax.xml.bind.ContextFinder.newInstance(ContextFinder.java:226)
at javax.xml.bind.ContextFinder.find(ContextFinder.java:441)
at javax.xml.bind.JAXBContext.newInstance(JAXBContext.java:641)
at javax.xml.bind.JAXBContext.newInstance(JAXBContext.java:584)
at org.hibernate.boot.cfgxml.internal.JaxbCfgProcessor.unmarshal(JaxbCfgProcessor.java:122)
... 17 more
Caused by: java.lang.ClassNotFoundException: com/sun/xml/bind/v2/ContextFactory
at java.lang.Class.forName0(Native Method)
at java.lang.Class.forName(Class.java:264)
at javax.xml.bind.ContextFinder.safeLoadClass(ContextFinder.java:577)
at javax.xml.bind.ContextFinder.newInstance(ContextFinder.java:224)
... 21 more
Все, что я смог найти об этой ошибке, это то, что JaxB не предоставляется для Java > 8, но я использую Java 8, и он отлично работает, если я просто создаю его при запуске приложения.
1 ответ
Я считаю, что это своего рода конфликт классового пути. В Java 8 следующий код не работает сClassNotFoundException: com/sun/xml/bind/v2/ContextFactory
, как указано в вопросе:
public class MyService {
@Get("/start")
public HttpResponse start() throws Exception {
final StandardServiceRegistryBuilder registryBuilder =
new StandardServiceRegistryBuilder().configure();
...
}
}
Однако проблема исчезнет после обновления до более новой версии Java, такой как Java 11.
К счастью, проблему можно обойти, явно указав загрузчик класса контекста:
@Get("/start")
public HttpResponse start() throws Exception {
Thread.currentThread().setContextClassLoader(MyService.class.getClassLoader());
// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
final StandardServiceRegistryBuilder registryBuilder =
new StandardServiceRegistryBuilder().configure();
...
}