我对构建其余的API应用程序非常陌生。在这里,我试图使一个简单的方法返回模型对象的响应xml Message
这是我的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();
}
}
现在运行此命令并尝试访问URL http://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
中添加依赖项:jersey return 500 when trying return an XML
我建议,我将模型对象的字段的修饰符设为私有。(使用的默认XML访问器类型jaxb可以在私有字段上工作)
任何人都可以提出建议吗?
经过多次尝试解决这个问题,我得到了解决方案:
关于在jaxb中导致java.lang.NullPointerException
与handleClassCastException
产生关系的一些背景知识,可以在https://github.com/javaee/jaxb-v2/issues/863处发现,并且发生类似的情况。
所以基本上,一次有多个用于JAXB的API的副本。我的应用程序将Jakarta API作为球衣依赖项的一部分(Jakarta现在是Java EE的标准API),但直到JDK 8为止,JAXB还是classpath中Java扩展api的一部分(javax支持)。我在使用jakarta api导入JAXB批注和类,但是也在使用jdk 8进行编译,因此引起了问题。来到jersey时,我正在使用Jersey 3.0.0的pom依赖关系实现REST API(这仍然是“太新了”,不能视为稳定版本)
如果您要坚持使用JDK 8,可以在所有的api和导入中使用javax,并且将pom中的jersey版本降级(我使用的是2.3.1)。
请注意,更高的球衣版本-3.0.0仅与jakarta API捆绑在一起,在这种情况下,您将把导入从javax转移到jakarta。
第二个选项是,如果您打算将JDK版本升级到1.8以上,则应该获得NoClassDefFoundError
,但是如果您仍在使用不大于10的JDK版本,则可以解决此问题。 Java EE和classpath的API默认情况下不包含它们。有一种启用它们的方法:How to resolve java.lang.NoClassDefFoundError: javax/xml/bind/JAXBException in Java 9
就我而言,我通过在pom中输入jaxb来提供对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>