为什么 Jersey 忽略我的 Jackson 注释?
Why is Jersey ignoring my Jackson annotations?
我正在使用 Jersey 和 Jackson 创建一个简单的 JSON API。
一些正在序列化的对象具有自定义枚举字段。默认情况下,这些枚举会根据枚举值转换为字符串——我希望枚举为 have slightly more complex serializations.
我在枚举中使用 Jackson 注释,但端点似乎忽略了它们。我一直在想方设法找出问题出在哪里,现在我求助于您。
枚举代码
package org.example.code;
import com.fasterxml.jackson.annotation.JsonFormat;
import com.fasterxml.jackson.annotation.JsonProperty;
@JsonFormat(shape= JsonFormat.Shape.OBJECT)
public enum ExampleEnum {
YES (1, "Yes indeed"),
NO (2, "No way buddy")
private final Integer code;
private final String description;
ExampleEnum(final Integer code, final String description) {
this.code = code;
this.description = description;
}
@JsonProperty("code")
public Integer getCode() {
return code;
}
@JsonProperty("description")
public String getDescription() {
return description;
}
}
API代码
package org.example.webservice.impl;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
import org.example.code.ExampleEnum;
@Path("/example")
public class ExampleService {
@GET
@Path("/test")
@Produces({MediaType.APPLICATION_JSON})
public ExampleEnum getExampleEnum() {
return ExampleEnum.YES;
}
}
当我调用端点 example/test
时,输出是 YES
我想要的是输出类似于 { code: 1, description: "Yes indeed" }
配置文件如下...
pom.xml
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>org.example</groupId>
<artifactId>api</artifactId>
<version>0.0.1</version>
<packaging>war</packaging>
<name>example API</name>
<url>http://example.org</url>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>
<dependencies>
<dependency>
<groupId>javax.ws.rs</groupId>
<artifactId>javax.ws.rs-api</artifactId>
<version>2.0.1</version>
</dependency>
<dependency>
<groupId>javax.annotation</groupId>
<artifactId>javax.annotation-api</artifactId>
<version>1.2</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-databind</artifactId>
<version>2.6.0</version>
</dependency>
<dependency>
<groupId>org.glassfish.jersey.containers</groupId>
<artifactId>jersey-container-servlet-core</artifactId>
<version>2.18</version>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.3</version>
<configuration>
<source>1.7</source>
<target>1.7</target>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.tomcat.maven</groupId>
<artifactId>tomcat7-maven-plugin</artifactId>
<version>2.2</version>
<configuration>
<url>http://localhost:8080/manager/text</url>
<server>TomcatServer</server>
<path>/example</path>
</configuration>
</plugin>
</plugins>
</build>
</project>
web.xml
<?xml version="1.0" encoding="UTF-8"?>
<web-app version="3.0"
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_3_0.xsd">
<display-name></display-name>
<servlet>
<display-name>Example Servlet</display-name>
<servlet-name>Example Servlet</servlet-name>
<servlet-class>org.glassfish.jersey.servlet.ServletContainer</servlet-class>
<init-param>
<param-name>jersey.config.server.provider.packages</param-name>
<param-value>org.example.webservice.impl</param-value>
</init-param>
<init-param>
<param-name>com.sun.jersey.api.json.POJOMappingFeature</param-name>
<param-value>true</param-value>
</init-param>
<init-param>
<param-name>jersey.config.server.provider.classnames</param-name>
<param-value>org.glassfish.jersey.filter.LoggingFilter;org.glassfish.jersey.media.multipart.MultiPartFeature</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>Example Servlet</servlet-name>
<url-pattern>/services/*</url-pattern>
</servlet-mapping>
<welcome-file-list>
<welcome-file>index.jsp</welcome-file>
</welcome-file-list>
</web-app>
有几件事:
com.sun.jersey.api.json.POJOMappingFeature
用于 Jersey 1.x,因此您可以去掉它。
MOXy 是 Glassfish 中的默认提供程序,因此如果您想使用 Jackson,则需要禁用 MOXy。您可以通过添加 <init-param>
<init-param>
<param-name>jersey.config.server.disableMoxyJson</param-name>
<param-value>true</param-value>
</init-param>
如果您的配置使用 ResourceConfig
,则可以使用 属性
public class JerseyConfig extends ResourceConfig {
public JerseyConfig() {
property(ServerProperties.MOXY_JSON_FEATURE_DISABLE, true);
}
}
常量值其实就是jersey.config.server.disableMoxyJson
.
然后添加 Jackson 提供商
<dependency>
<groupId>org.glassfish.jersey.media</groupId>
<artifactId>jersey-media-json-jackson</artifactId>
<version>${jersey2.version}</version>
<scope>provided</scope>
</dependency>
Jersey 版本应与您在项目中使用的任何版本相匹配。但是,需要注意的一件非常重要的事情是,如果您使用的是 Glassfish,则应该将所有 Jersey 依赖项放在 <scope>provided</scope>
中,因为服务器的库中已经有这些 jar。您不希望您的项目依赖项与服务器中已有的 jar 冲突。
接下来您需要在您的应用程序中注册 Jackson 提供程序。如果您使用 web.xml,您可以将 JacksonFeature
添加到 ...classnames
属性
的列表中
<init-param>
<param-name>jersey.config.server.provider.classnames</param-name>
<param-value>
org.glassfish.jersey.filter.LoggingFilter,
org.glassfish.jersey.media.multipart.MultiPartFeature,
org.glassfish.jersey.jackson.JacksonFeature
</param-value>
</init-param>
如果您的配置使用 ResourceConfig
,那么您只需注册 JacksonFeature
public class JerseyConfig extends ResourceConfig {
public JerseyConfig() {
register(JacksonFeature.class);
}
}
与我可能指出的这个问题无关的另一件事是,Glassfish 4 已经有一个 Jersey 实现,这是一个非常旧的 2.x 版本,可能是 2.0。当您添加运行时 Jersey 依赖项时,它们可能会发生冲突。我会做以下两件事之一,将所有 Jersey 依赖项放在 provided
<scope>
中,或者如果您对更高版本的功能有要求,您可能需要研究在 Glassfish 中更新 Jersey 版本。看看 Updating Jersey 2 in GlassFish 4
Glassfish 本身有 jackson jar,它可能与您在 pom.xml 中声明的显式 jackson jar 有冲突。
在 maven 网站上,对于每个版本的 jar,您选择的 jar 中几乎没有其他编译时依赖 jar。
所以这些也可能与您在 pom.xml
中声明的 jar 冲突
我也删除了
<dependency>
<groupId>org.glassfish.jersey.media</groupId>
<artifactId>jersey-media-json-jackson</artifactId>
<version>2.24</version>
</dependency>
来自 pom.xml
对于使用应用程序 class 而不是使用 web.xml 的 Jersey 2.2x 用户,这对我有用
1.在ResourceConfig中,
@ApplicationPath("rest")
public class JerseyApplication extends ResourceConfig {
public JerseyApplication() {
packages("com.example.rest");
register(JacksonFeature.class);
}
}
或者,如果您使用的是应用程序 class,
public class MyApplication extends Application {
public Set<Class<?>> getClasses() {
final Set<Class<?>> classes = new HashSet<Class<?>>();
.
.
classes.add(JacksonFeature.class);
return classes;
}
}
2.在pom.xml中添加这个依赖,
<dependency>
<groupId>org.glassfish.jersey.media</groupId>
<artifactId>jersey-media-json-jackson</artifactId>
<version>2.26</version>
</dependency>
3. 确保在 getter 和 setter 上也有注释。在我的例子中,@JsonIgnore 注释仅在我在该 class 变量的 getter 和 setter 上使用它时才起作用。
我正在使用 Jersey 和 Jackson 创建一个简单的 JSON API。
一些正在序列化的对象具有自定义枚举字段。默认情况下,这些枚举会根据枚举值转换为字符串——我希望枚举为 have slightly more complex serializations.
我在枚举中使用 Jackson 注释,但端点似乎忽略了它们。我一直在想方设法找出问题出在哪里,现在我求助于您。
枚举代码
package org.example.code;
import com.fasterxml.jackson.annotation.JsonFormat;
import com.fasterxml.jackson.annotation.JsonProperty;
@JsonFormat(shape= JsonFormat.Shape.OBJECT)
public enum ExampleEnum {
YES (1, "Yes indeed"),
NO (2, "No way buddy")
private final Integer code;
private final String description;
ExampleEnum(final Integer code, final String description) {
this.code = code;
this.description = description;
}
@JsonProperty("code")
public Integer getCode() {
return code;
}
@JsonProperty("description")
public String getDescription() {
return description;
}
}
API代码
package org.example.webservice.impl;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
import org.example.code.ExampleEnum;
@Path("/example")
public class ExampleService {
@GET
@Path("/test")
@Produces({MediaType.APPLICATION_JSON})
public ExampleEnum getExampleEnum() {
return ExampleEnum.YES;
}
}
当我调用端点 example/test
时,输出是 YES
我想要的是输出类似于 { code: 1, description: "Yes indeed" }
配置文件如下...
pom.xml
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>org.example</groupId>
<artifactId>api</artifactId>
<version>0.0.1</version>
<packaging>war</packaging>
<name>example API</name>
<url>http://example.org</url>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>
<dependencies>
<dependency>
<groupId>javax.ws.rs</groupId>
<artifactId>javax.ws.rs-api</artifactId>
<version>2.0.1</version>
</dependency>
<dependency>
<groupId>javax.annotation</groupId>
<artifactId>javax.annotation-api</artifactId>
<version>1.2</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-databind</artifactId>
<version>2.6.0</version>
</dependency>
<dependency>
<groupId>org.glassfish.jersey.containers</groupId>
<artifactId>jersey-container-servlet-core</artifactId>
<version>2.18</version>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.3</version>
<configuration>
<source>1.7</source>
<target>1.7</target>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.tomcat.maven</groupId>
<artifactId>tomcat7-maven-plugin</artifactId>
<version>2.2</version>
<configuration>
<url>http://localhost:8080/manager/text</url>
<server>TomcatServer</server>
<path>/example</path>
</configuration>
</plugin>
</plugins>
</build>
</project>
web.xml
<?xml version="1.0" encoding="UTF-8"?>
<web-app version="3.0"
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_3_0.xsd">
<display-name></display-name>
<servlet>
<display-name>Example Servlet</display-name>
<servlet-name>Example Servlet</servlet-name>
<servlet-class>org.glassfish.jersey.servlet.ServletContainer</servlet-class>
<init-param>
<param-name>jersey.config.server.provider.packages</param-name>
<param-value>org.example.webservice.impl</param-value>
</init-param>
<init-param>
<param-name>com.sun.jersey.api.json.POJOMappingFeature</param-name>
<param-value>true</param-value>
</init-param>
<init-param>
<param-name>jersey.config.server.provider.classnames</param-name>
<param-value>org.glassfish.jersey.filter.LoggingFilter;org.glassfish.jersey.media.multipart.MultiPartFeature</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>Example Servlet</servlet-name>
<url-pattern>/services/*</url-pattern>
</servlet-mapping>
<welcome-file-list>
<welcome-file>index.jsp</welcome-file>
</welcome-file-list>
</web-app>
有几件事:
com.sun.jersey.api.json.POJOMappingFeature
用于 Jersey 1.x,因此您可以去掉它。MOXy 是 Glassfish 中的默认提供程序,因此如果您想使用 Jackson,则需要禁用 MOXy。您可以通过添加
<init-param>
<init-param> <param-name>jersey.config.server.disableMoxyJson</param-name> <param-value>true</param-value> </init-param>
如果您的配置使用
ResourceConfig
,则可以使用 属性public class JerseyConfig extends ResourceConfig { public JerseyConfig() { property(ServerProperties.MOXY_JSON_FEATURE_DISABLE, true); } }
常量值其实就是
jersey.config.server.disableMoxyJson
.然后添加 Jackson 提供商
<dependency> <groupId>org.glassfish.jersey.media</groupId> <artifactId>jersey-media-json-jackson</artifactId> <version>${jersey2.version}</version> <scope>provided</scope> </dependency>
Jersey 版本应与您在项目中使用的任何版本相匹配。但是,需要注意的一件非常重要的事情是,如果您使用的是 Glassfish,则应该将所有 Jersey 依赖项放在
<scope>provided</scope>
中,因为服务器的库中已经有这些 jar。您不希望您的项目依赖项与服务器中已有的 jar 冲突。接下来您需要在您的应用程序中注册 Jackson 提供程序。如果您使用 web.xml,您可以将
的列表中JacksonFeature
添加到...classnames
属性<init-param> <param-name>jersey.config.server.provider.classnames</param-name> <param-value> org.glassfish.jersey.filter.LoggingFilter, org.glassfish.jersey.media.multipart.MultiPartFeature, org.glassfish.jersey.jackson.JacksonFeature </param-value> </init-param>
如果您的配置使用
ResourceConfig
,那么您只需注册JacksonFeature
public class JerseyConfig extends ResourceConfig { public JerseyConfig() { register(JacksonFeature.class); } }
与我可能指出的这个问题无关的另一件事是,Glassfish 4 已经有一个 Jersey 实现,这是一个非常旧的 2.x 版本,可能是 2.0。当您添加运行时 Jersey 依赖项时,它们可能会发生冲突。我会做以下两件事之一,将所有 Jersey 依赖项放在 provided
<scope>
中,或者如果您对更高版本的功能有要求,您可能需要研究在 Glassfish 中更新 Jersey 版本。看看 Updating Jersey 2 in GlassFish 4
Glassfish 本身有 jackson jar,它可能与您在 pom.xml 中声明的显式 jackson jar 有冲突。
在 maven 网站上,对于每个版本的 jar,您选择的 jar 中几乎没有其他编译时依赖 jar。
所以这些也可能与您在 pom.xml
中声明的 jar 冲突我也删除了
<dependency>
<groupId>org.glassfish.jersey.media</groupId>
<artifactId>jersey-media-json-jackson</artifactId>
<version>2.24</version>
</dependency>
来自 pom.xml
对于使用应用程序 class 而不是使用 web.xml 的 Jersey 2.2x 用户,这对我有用
1.在ResourceConfig中,
@ApplicationPath("rest")
public class JerseyApplication extends ResourceConfig {
public JerseyApplication() {
packages("com.example.rest");
register(JacksonFeature.class);
}
}
或者,如果您使用的是应用程序 class,
public class MyApplication extends Application {
public Set<Class<?>> getClasses() {
final Set<Class<?>> classes = new HashSet<Class<?>>();
.
.
classes.add(JacksonFeature.class);
return classes;
}
}
2.在pom.xml中添加这个依赖,
<dependency>
<groupId>org.glassfish.jersey.media</groupId>
<artifactId>jersey-media-json-jackson</artifactId>
<version>2.26</version>
</dependency>
3. 确保在 getter 和 setter 上也有注释。在我的例子中,@JsonIgnore 注释仅在我在该 class 变量的 getter 和 setter 上使用它时才起作用。