泽西岛:具有 1 个元素的 Json 数组被序列化为对象

Posted

技术标签:

【中文标题】泽西岛:具有 1 个元素的 Json 数组被序列化为对象【英文标题】:Jersey: Json array with 1 element is serialized as object 【发布时间】:2012-11-14 13:05:36 【问题描述】:

我正在使用 Jersey/Java 创建一个 REST 服务器,但我发现了一个奇怪的行为。

我在服务器上有一个方法,它以 Json 形式返回一组对象

@GET
@Path("/files")
@Produces(MediaType.APPLICATION_JSON)
public Object getFiles() throws Exception
    DatabaseManager db = new DatabaseManager();
    FileInfo[] result = db.getFiles();
    return result;

代码正确执行并将数据返回给客户端(jQuery ajax 调用)。 问题是如果“结果”数组有一个或多个元素,返回数据的格式会发生变化。

一个元素的响应:

"fileInfo":"fileName":"weather.arff","id":"10"

包含两个元素的响应:

"fileInfo":["fileName":"weather.arff","id":"10","fileName":"supermarket.arff","id":"11"]

如您所见,在第一种情况下,返回对象的“fileInfo”属性的值是一个对象,而在第二种情况下,该值是一个数组。 我究竟做错了什么?第一种情况不应该返回这样的东西吗:

"fileInfo":["fileName":"weather.arff","id":"10"]

即里面只有一个对象的数组?

我知道我可以在客户端检测到这一点,但这似乎是一个非常丑陋的 hack。

感谢您的宝贵时间。

【问题讨论】:

【参考方案1】:

我最终使用了 Jackson,在泽西官方文档 (http://jersey.java.net/nonav/documentation/latest/user-guide.html#json.pojo.approach.section) 中也有描述。/nonav/documentation/latest/user-guide.html#json.pojo.approach.section p>

我之前尝试过,但没有成功,因为我的项目的构建路径中没有 jackson jar(根据文档,我认为它已内置到 jersey 的核心库中)。

我刚刚添加了 jackson-all.jar 文件 (http://wiki.fasterxml.com/JacksonDownload) 并在配置中启用了 POJO 支持

    <init-param>
          <param-name>com.sun.jersey.api.json.POJOMappingFeature</param-name>
          <param-value>true</param-value>
    </init-param>

瞧!

【讨论】:

这回答了我的问题,而没有修改我的代码中的任何内容。谢谢。【参考方案2】:

如果您使用 JAXB 构建 JSON 结果,您可以配置 Jersey JSON 处理器以获得更重要的 JSON 格式。

jersey official document有详细配置:

要实现更重要的 JSON 格式更改,您需要配置 Jersey JSON 处理器本身。可以在 JSONConfiguration 实例上设置各种配置选项。然后可以进一步使用该实例来创建 JSONConfigurated JSONJAXBContext,作为该区域的主要配置点。要将您的专用 JSONJAXBContext 传递给 Jersey,您最终需要实现一个 JAXBContext ContextResolver:

    @Provider
public class JAXBContextResolver implements ContextResolver<JAXBContext> 
    private final JAXBContext context;
    private final Set<Class> types;
    private Class[] ctypes =  FileInfo.class; //your pojo class
    public JAXBContextResolver() throws Exception 
        this.types = new HashSet(Arrays.asList(ctypes));
        this.context = new JSONJAXBContext(JSONConfiguration.natural().build(),
                ctypes); //json configuration
    

    @Override
    public JAXBContext getContext(Class<?> objectType) 
        return (types.contains(objectType)) ? context : null;
    

【讨论】:

添加此解析器后响应将如何格式化,所以我需要在某处告诉代码使用此解析器?【参考方案3】:

还请查看以下答案,它为我解决了这个问题:

How can I customize serialization of a list of JAXB objects to JSON?

【讨论】:

【参考方案4】:

我正在使用 cxf,这是我的 applicationContext.xml 以强制 JSON 中的数组:

<jaxrs:server id="myService" serviceName="MyService"
address="/mysvc">
<jaxrs:serviceBeans>
    <ref bean="myServiceImpl"/>
</jaxrs:serviceBeans>
<jaxrs:providers>
    <bean class="org.apache.cxf.jaxrs.provider.json.JSONProvider">
   <property name="dropRootElement" value="true" />
   <property name="supportUnwrapped" value="true" />
   <property name="namespaceMap">
      <map>
        <entry key="http://example.com/myservice" value=""/>
      </map>
   </property>
   <property name="arrayKeys">
      <list>
    <value>fileInfo</value>
      </list>
   </property>                          
    </bean>
</jaxrs:providers>
</jaxrs:server>

【讨论】:

这显然是使用 spring 配置的唯一方法。这不是最佳选择,但就我而言,这是唯一的方法。【参考方案5】:

您可以使用 Jettison(与 Jersey 一起提供)并使用 JSONObjectJSONArray 作为返回值来准备您想要的结构。 它们位于jettison-1.3.2.jar 的包org.codehaus.jettison.json 中,这是jerysey-json 的传递依赖项

【讨论】:

JSONArray 类在哪里定义?我必须从 json.org 添加 json 库吗?【参考方案6】:

您也可以尝试 Genson 库 http://code.google.com/p/genson/。它与 jersey 很好地集成,只需将 jar 放入您的类路径中,一切都会正常工作。它不需要你编写额外的代码,它应该像你现在拥有的那样工作,但没有任何奇怪的结果。

【讨论】:

【参考方案7】:

我苦苦挣扎,找到了这个简单的解决方案

在你的 pom.xml 中:

<dependency>
    <groupId>org.codehaus.jackson</groupId>
    <artifactId>jackson-jaxrs</artifactId>
    <version>1.9.13</version>
</dependency>
<dependency>
    <groupId>org.codehaus.jackson</groupId>
    <artifactId>jackson-xc</artifactId>
    <version>1.9.13</version>
</dependency>

在您的 web.xml 中:

<servlet-class>com.sun.jersey.spi.container.servlet.ServletContainer</servlet-class>
<init-param>
    <param-name>com.sun.jersey.config.property.packages</param-name>
    <param-value>com.other-packages;org.codehaus.jackson.jaxrs</param-value>
</init-param>

【讨论】:

【参考方案8】:

将 Array 转换为 ArrayList 就可以满足这里的要求。 我遇到过类似的矛盾问题,如果是单个元素,我必须返回 Json 数组对象而不是列表。

我在下面的注释的帮助下完成了我的工作-

@JsonFormat(with = JsonFormat.Feature.WRITE_SINGLE_ELEM_ARRAYS_UNWRAPPED)。 以下是 JSON Pojo 类的示例:

import java.util.List;

import com.fasterxml.jackson.annotation.JsonFormat;
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import com.fasterxml.jackson.annotation.JsonProperty;

@JsonIgnoreProperties(ignoreUnknown = true)
public class TAResponseMapper 

    @JsonProperty("Response")
    @JsonFormat(with = JsonFormat.Feature.WRITE_SINGLE_ELEM_ARRAYS_UNWRAPPED) 
    private List<TAResponse> responses;

    public List<TAResponse> getResponses() 
        return responses;
    

    public void setResponses(List<TAResponse> responses) 
        this.responses = responses;
    


【讨论】:

以上是关于泽西岛:具有 1 个元素的 Json 数组被序列化为对象的主要内容,如果未能解决你的问题,请参考以下文章

反序列化具有一个元素的列表,如果列表大小为 1,则直接作为元素本身(Json 格式)

泽西岛的 JSON ArrayList

无法将当前 JSON 数组(例如 [1,2,3])反序列化为具有复杂和嵌套对象的类型

在训练 SVM 对图像进行分类时设置具有序列错误的数组元素

LeetCode刷题8——两数之和

最多具有K个元素的最大连续子数组