java.lang.NullPointerException at jakarta.xml.bind.ContextFi



我对构建rest API应用程序非常陌生。在这里,我试图让一个简单的方法返回模型对象Message的响应xml

这是我的web.xml

<?xml version="1.0" encoding="UTF-8"?>
<!-- This web.xml file is not required when using Servlet 3.0 container,
see implementation details http://jersey.java.net/nonav/documentation/latest/jax-rs.html -->
<web-app version="2.5" xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
<servlet>
<servlet-name>Jersey Web Application</servlet-name>
<servlet-class>org.glassfish.jersey.servlet.ServletContainer</servlet-class>
<init-param>
<param-name>jersey.config.server.provider.packages</param-name>
<param-value>com.rest.messenger.service</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>Jersey Web Application</servlet-name>
<url-pattern>/webapi/*</url-pattern>
</servlet-mapping>
</web-app>

我有一个服务类,它返回一个硬编码的Message对象列表。

package com.rest.messenger.service;
import java.util.ArrayList;
import java.util.List;
import com.rest.messenger.model.Message;
public class MessageService {

public List<Message> getAllMessages(){
Message m1 = new Message(1L, "message1", "mt");
Message m2 = new Message( 2L, "message 2", "pt");
List<Message> messages = new ArrayList<Message>();
messages.add(m1);
messages.add(m2);
return messages;
}
}

Message对象定义如下(这里我使用@XmlRootElement注释让jaxb将其转换为xml,因为我希望资源返回xml响应(:

package com.rest.messenger.model;
import java.util.Date;
import jakarta.xml.bind.annotation.XmlRootElement;
@XmlRootElement
public class Message {
private long id;
private String message;
private String author;
private Date created;
public Message() {
}
public Message(long id, String message, String author) {
super();
this.id = id;
this.message = message;
this.author = author;
this.created = new Date();
}
public long getId() {
return id;
}
public void setId(long id) {
this.id = id;
}
public String getMessage() {
return message;
}
public void setMessage(String message) {
this.message = message;
}
public String getAuthor() {
return author;
}
public void setAuthor(String author) {
this.author = author;
}
public Date getCreated() {
return created;
}
public void setCreated(Date created) {
this.created = created;
}
}

我将传入的GET请求映射到一个资源,如下所示:

import jakarta.ws.rs.GET;
import jakarta.ws.rs.Path;
import jakarta.ws.rs.Produces;
import jakarta.ws.rs.core.MediaType;
@Path("/messages")
public class MessageResource {
public MessageService messageService = new MessageService();
@GET
@Produces(MediaType.APPLICATION_XML)
public List<Message> getMessages() {
return messageService.getAllMessages();
}
}

现在,当我运行这个并尝试访问URLhttp://localhost:8080/messenger.service/webapi/messages上的资源时

我得到以下内部服务器错误:

May 06, 2020 1:28:14 PM org.apache.catalina.core.StandardWrapperValve invoke
SEVERE: Servlet.service() for servlet [Jersey Web Application] in context with path [/messenger.service] threw exception [java.lang.NullPointerException] with root cause
java.lang.NullPointerException
at jakarta.xml.bind.ContextFinder.handleClassCastException(ContextFinder.java:114)
at jakarta.xml.bind.ContextFinder.newInstance(ContextFinder.java:254)
at jakarta.xml.bind.ContextFinder.newInstance(ContextFinder.java:240)
at jakarta.xml.bind.ContextFinder.find(ContextFinder.java:375)
at jakarta.xml.bind.JAXBContext.newInstance(JAXBContext.java:691)
at jakarta.xml.bind.JAXBContext.newInstance(JAXBContext.java:632)
at org.glassfish.jersey.jaxb.internal.AbstractJaxbProvider.getStoredJaxbContext(AbstractJaxbProvider.java:288)
at org.glassfish.jersey.jaxb.internal.AbstractJaxbProvider.getJAXBContext(AbstractJaxbProvider.java:273)
at org.glassfish.jersey.jaxb.internal.AbstractJaxbProvider.getMarshaller(AbstractJaxbProvider.java:240)
at org.glassfish.jersey.jaxb.internal.AbstractJaxbProvider.getMarshaller(AbstractJaxbProvider.java:207)
at org.glassfish.jersey.jaxb.internal.AbstractCollectionJaxbProvider.writeTo(AbstractCollectionJaxbProvider.java:243)
at org.glassfish.jersey.message.internal.WriterInterceptorExecutor$TerminalWriterInterceptor.invokeWriteTo(WriterInterceptorExecutor.java:242)

我遵循的教程使用了javax,我相信它现在已经迁移到jakarta。不确定是否与@XmlRootElement有关我还在pom.xml中添加了依赖项,建议如下:尝试返回XML 时球衣返回500

正如建议的那样,我使用了模型对象的字段的修饰符作为私有字段。(jaxb使用的默认XML访问器类型能够处理私有字段(

有人能对此提出建议吗?

经过多次尝试,我得到了解决方案:

可以在以下位置找到导致java.lang.NullPointerException的一些背景场景:在使用JAXB时堆栈跟踪显示handleClassCastExceptionhttps://github.com/javaee/jaxb-v2/issues/863类似的事情正在发生。

因此,基本上,JAXB同时有多个API副本。我的应用程序将Jakarta API作为jersey依赖项的一部分(Jakarta现在是Java EE的标准API(,但在JDK 8之前,JAXB也是类路径中Java扩展API的一部分。我使用jakartaapi导入JAXB注释和类,但也使用jdk8进行编译,因此导致了这个问题。在使用jersey时,我使用jersey 3.0.0的pom依赖性实现了REST API(它仍然"太新",不能被视为稳定版本(

解决方案可能是对所有api和导入使用javax,如果您要坚持使用JDK 8,并在pom中降级jersey版本(我使用了2.3.1([或者外部JAR版本,如果您不使用maven(

请注意,更高版本的jersey 3.0.0仅与jakarta API捆绑在一起,在这种情况下,您必须将导入从javax转移到jakarta。

第二种选择是,如果您打算升级高于1.8的JDK版本,则应该获得NoClassDefFoundError,但如果您仍然使用不超过10的JDK,则可以解决此问题。Jaxb API现在被认为是Java EE的API,默认情况下类路径不包含它们。然而,有一种方法可以使它们发挥作用:如何在java 9 中解析java.lang.NoClassDefFoundError:javax/xml/bind/JJAXBException

在我的案例中,我通过在pom.xml中输入来为JAXB实现类提供支持

总之,我做了以下几点:

  • 我继续使用JDK10和球衣2.3.1
  • 通过在POM中添加依赖项提供了JAXB的支持:
<dependency>
<groupId>javax.xml.bind</groupId>
<artifactId>jaxb-api</artifactId>
<version>2.3.1</version>
</dependency>
<dependency>
<groupId>org.glassfish.jaxb</groupId>
<artifactId>jaxb-runtime</artifactId>
<version>2.3.1</version>
<scope>runtime</scope>
</dependency>

相关内容

最新更新