JAXB用于为JSON或XML自然返回的列表

Sco*_*ott 8 java xml json moxy

我正在使用MOXy和Jersey实现一个RESTful API,并希望自然地为JSON和XML返回列表,我的意思是XML包含整个集合的元素标记以及集合项,而JSON包含仅用于集合的标记.

例如,我想返回一个"组织"资源,其中包含设施和位置的嵌套列表.作为XML:

<organisation>
  <id>1</id>
  <name>XYZ</name>
  <facilities>
    <facility>
      <id>1</id>
      <text>Telephone</text>
    </facility>
    <facility>
      <id>3</id>
      <text>Whiteboard</text>
    </facility>
  </facilities>
  <locations>
    <location>
      <id>1</id>
      <kind>REGION</kind>
      <name>London</name>
    </location>
    <location>
      <id>2</id>
      <kind>REGION</kind>
      <name>Manchester</name>
    </location>
  </locations>
</organisation>
Run Code Online (Sandbox Code Playgroud)

而作为JSON:

{
  "id": 1,
  "name": "XYZ",
  "facilities": [
    {
      "id": 1,
      "text": "Telephone"
    },
    {
      "id": 3,
      "text": "Whiteboard"
    }
  ],
  "locations": [
    {
      "id": 1,
      "kind": "REGION",
      "name": "London"
    },
    {
      "id": 2,
      "kind": "REGION",
      "name": "Manchester"
    }
  ]
}
Run Code Online (Sandbox Code Playgroud)

不幸的是,我似乎无法获得允许我为XML和JSON返回这样的输出的代码库.如果我使用一个类来包装嵌套列表,那么XML看起来是正确的但不是JSON(参见下面的"ExternalFacilities").如果我将嵌套列表定义为ArrayList子类,则JSON显示正确但不是XML(请参阅下面的"ExternalLocations").

XML示例显示"设施"列表正确但不是"位置"

看到没有包含"位置"列表的XML元素(就像"设施"一样),并且每个位置都有一个复数元素名称.

<organisation>
  <id>1</id>
  <name>XYZ</name>
  <facilities>
    <facility>
      <id>1</id>
      <text>Telephone</text>
    </facility>
    <facility>
        <id>3</id>
        <text>Whiteboard</text>
    </facility>
  </facilities>
  <locations>
    <id>1</id>
    <kind>REGION</kind>
    <name>London</name>
  </locations>
  <locations>
    <id>2</id>
    <kind>REGION</kind>
    <name>Manchester</name>
  </locations>
</organisation>
Run Code Online (Sandbox Code Playgroud)

JSON样本显示"位置"列表正确但不是"设施"

看到"facility"列表是一个包含JSON数组的JSON对象,而我只想要JSON数组(带有复数元素名称).

{
  "id": 1,
  "name": "XYZ",
  "facilities": {
    "facility": [
        {
            "id": 1,
            "text": "Telephone"
        },
        {
            "id": 3,
            "text": "Whiteboard"
        }
    ]
  },
  "locations": [
    {
      "id": 1,
      "kind": "REGION",
      "name": "London"
    },
    {
      "id": 2,
      "kind": "REGION",
      "name": "Manchester"
    }
  ]
}
Run Code Online (Sandbox Code Playgroud)

上面的示例使用相同的代码生成,只需更改Accept HTTP标头以使Jersey返回JSON而不是XML.以下是课程的摘录:

ExternalOrganisation.java

@XmlRootElement(name="organisation")
@XmlAccessorType(XmlAccessType.FIELD)
public class ExternalOrganisation {

   private String             name;
   private ExternalFacilities facilities;
   private ExternalLocations  locations;

   ...

}
Run Code Online (Sandbox Code Playgroud)

ExternalFacilities.java

@XmlRootElement(name="facilities")
public class ExternalFacilities {
  @XmlElementRef
  private List<ExternalFacility> list;
  ...
}
Run Code Online (Sandbox Code Playgroud)

ExternalLocations.java

@XmlRootElement(name="locations")
public class ExternalLocations extends ArrayList<ExternalLocation> {
  ...
}
Run Code Online (Sandbox Code Playgroud)

ExternalFacility.java

@XmlRootElement(name="facility")
@XmlType(propOrder={"id", "uri", "kind", "text"})
public class ExternalFacility extends ExternalBase {
    // id inherited
    private String text;
    ....
}
Run Code Online (Sandbox Code Playgroud)

ExternalLocation.java

@XmlRootElement(name="location")
@XmlType(propOrder={"id", "kind", "name"})
public class ExternalLocation extends ExternalBase {
    // id inherited
    @XmlElement private LocationKind kind;
    @XmlElement private String       name;
    ...
}
Run Code Online (Sandbox Code Playgroud)

似乎最初类似于这个问题,但我不是想在我的列表中混合对象的类型.

bdo*_*han 4

您可以执行以下操作来获取所需的 XML 和 JSON 表示形式:

步骤#1 - 杠杆@XMLElementWrapper

代替:

@XmlRootElement(name="organisation")
@XmlAccessorType(XmlAccessType.FIELD)
public class ExternalOrganisation {

   private String             name;
   private ExternalFacilities facilities;
   private ExternalLocations  locations;

   ...

}
Run Code Online (Sandbox Code Playgroud)

您可以执行以下操作@XmlElementWrapper(请参阅: http://blog.bdoughan.com/2010/09/jaxb-collection-properties.html):

@XmlRootElement(name="organisation")
@XmlAccessorType(XmlAccessType.FIELD)
public class ExternalOrganisation {

   private String             name;

   @XmlElementWrapper
   @XmlElementRef
   private List<ExternalFacility> facilities;

   @XmlElementWrapper
   @XmlElementRef
   private List<ExternalLocation>  locations;

   ...

}
Run Code Online (Sandbox Code Playgroud)

步骤 #2 - 利用 MOXy 的包装器作为数组名称属性

通过将包装器指定为数组名称属性,MOXy 将使用来自的值@XmlElementWrapper作为 JSON 数组名称。

import java.util.*;
import javax.ws.rs.core.Application;
import org.eclipse.persistence.jaxb.rs.MOXyJsonProvider;

public class YourApplication  extends Application {

    @Override
    public Set<Class<?>> getClasses() {
        HashSet<Class<?>> set = new HashSet<Class<?>>(1);
        set.add(YourService.class);
        return set;
    }

    @Override
    public Set<Object> getSingletons() {
        MOXyJsonProvider moxyJsonProvider = new MOXyJsonProvider();
        moxyJsonProvider.setWrapperAsArrayName(true);

        HashSet<Object> set = new HashSet<Object>(1);
        set.add(moxyJsonProvider);
        return set;
    }

} 
Run Code Online (Sandbox Code Playgroud)

了解更多信息

  • 我之前读过您的博客文章,但没有遇到这些内容。我已经尝试过并让它工作,但是我发现我必须指定`@XmlElementWrapper @XmlElement(name="facility")`,而不是使用`@XmlElementWrapper @XmlElementRef`,以便获取XML和 JSON 对。也许我还做错了什么,但我对这个解决方案很满意。我在 Stack Overflow 上阅读了您的很多答案,因此不仅感谢这个答案,还感谢所有其他已证明内容丰富且有帮助的答案。 (2认同)