1

我正在创建我的第一个 Web 服务,所以可能是我遗漏了一些非常简单的东西。我在 Eclipse Kepler 中使用 Jersey 2.x 在 Tomcat 上创建了一个没有 Maven 的 Web 服务,它适用于没有参数的“@GET”请求(从浏览器和客户端应用程序测试),但我遇到了“@POST”问题(代码如下)。这实际上是一个过滤条件非常复杂的get请求。

@POST
@Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.APPLICATION_JSON)
public String getFilteredPictures(ArrayList<FilterOption> filters)
{
    PictureProvider provider = new PictureProvider();
    ArrayList<PictureInfo> pictures;
    try
    {
        pictures = provider.getPictures(filters);
        Gson gson = new Gson();
        return gson.toJson(pictures);
    }
    catch (SQLException e)
    {
        // TODO Auto-generated catch block
        e.printStackTrace();
        return null;
    }
}

我创建了一个虚拟客户端,只是为了查看上面的方法是否有效:

HttpClient httpclient = new DefaultHttpClient();
    Gson gson = new Gson();
    HttpPost request = new HttpPost(SERVICE_URI + picturesServiceEndPoint);
     //create dummy data
    ArrayList<FilterOption> filters = new ArrayList<>();
    ArrayList<String> options = new ArrayList<>();
    options.add("Black");
    filters.add(new FilterOption("Color", options));
    StringEntity postParam = StringEntity(gson.toJson(filters), "UTF-8");
    postParam.setContentType("application/json");

    request.setEntity(postParam);
    request.setHeader("Accept", "application/json");
    try
    {
        HttpResponse response = httpclient.execute(request);
        HttpEntity entity = response.getEntity();
        if (entity != null)
        {
            //obtain results..

        }
    }
    catch (ClientProtocolException e)
    {
        e.printStackTrace();
    }
    catch (IOException e)
    {
        e.printStackTrace();
    }

当我运行客户端时,服务器抛出以下异常“.MessageBodyProviderNotFoundException: MessageBodyReader not found for media type=application/json”: 在此处输入图像描述

我怀疑问题是它无法将 JSON 转换为我的 POJO 对象,所以我在 web.xml 中放置了一个 init 参数,但它没有效果。另外,我尝试只发送一个 FilterOption 对象,认为 ArrayList 太复杂了,但还是没有效果。

感谢您的时间:)

4

5 回答 5

1

Jersey JSON 支持作为一组扩展模块提供,其中每个模块都包含一个Feature需要注册到您的Configurable实例(客户端/服务器)中的实现。有多个框架支持 JSON 处理和/或 JSON 到 Java 的绑定。下面列出的模块通过将各个 JSON 框架集成到 Jersey 中来提供对 JSON 表示的支持。目前,Jersey 集成了以下模块来提供 JSON 支持:

  • MOXy - 自 Jersey 2.0 以来,通过 MOXy 支持 JSON 绑定是在 Jersey 应用程序中支持 JSON 绑定的默认和首选方式。当 JSON MOXy 模块位于类路径上时,Jersey 将自动发现该模块并通过 MOXy 在您的应用程序中无缝启用 JSON 绑定支持。(请参阅第 4.3 节“自动发现的功能”。)
  • 用于 JSON 处理的 Java API (JSON-P)
  • 杰克逊
  • 抛弃

有关更多信息,请阅读球衣文档的第 9 章。

莫西:

Moxy 是 json 媒体支持的建议方法。MOXy 媒体模块是您不需要在客户端/服务器中显式注册其功能 (MoxyJsonFeature) 的模块之一 可配置,因为当您将 jersey-media-moxy 模块添加到您的类路径时会自动发现和注册此功能.

要使用 MOXy 作为您的 JSON 提供程序,您需要将 jersey-media-moxy 模块添加到您的 pom.xml 文件中:

<dependency>
   <groupId>org.glassfish.jersey.media</groupId>
   <artifactId>jersey-media-moxy</artifactId>
   <version>2.15</version>
</dependency>

如果您不使用 Maven,请确保拥有所有需要的依赖项。请参阅jersey-media-moxy 依赖项
您需要将这些 jar 文件添加到您的项目中,以便通过 jersey-media-moxy 支持 json 媒体类型:

  • jersey-media-moxy-2.15.jar
  • org.eclipse.persistence.core-2.5.0-rc2.jar
  • org.eclipse.persistence.antlr-2.5.0.jar
  • org.eclipse.persistence.moxy-2.5.0.jar
  • 球衣实体过滤-2.3.1.jar

用法:

一些普通的类:

public class MyJAXBBean{

   private String name = "jack";
   private int id = 12;

   public MyJAXBBean() {

   }

   public String getName() {
       return this.name;
   }

   public void setName(String name) {
       this.name = name;
   }

   public int getId() {
       return this.id;
   }

   public void setId(int id) {
       this.id = id;
   }
}

还有一个用于运行球衣客户端示例的主类:

 public static void main(String[] args) {
    //ClientConfig cc = new ClientConfig().register(new JacksonFeature());
    Client client = ClientBuilder.newClient();
    WebTarget target = client.target("http://localhost:8084/myhost/test");

    Form form = new Form();
    form.param("x", "foo");
    form.param("y", "bar");

    MyJAXBBean bean;
    bean = target.request(MediaType.APPLICATION_JSON_TYPE)
            .post(Entity.entity(form, MediaType.APPLICATION_FORM_URLENCODED_TYPE),
                    MyJAXBBean.class);
    System.out.println(bean);
}

server( http://localhost:8084/myhost/test) 的 json 响应必须采用以下格式:

 {"name":"haleh", "id":3}
于 2015-02-02T06:03:21.963 回答
1

通过实现 MessageBodyWriter 和 MessageBodyReader 2 个接口,您仍然可以保留方法的签名、媒体类型,并仍然使用 GSON 来编组/解组。

我手头没有自己项目的代码示例,但以下看起来不错: http ://eclipsesource.com/blogs/2012/11/02/integrating-gson-into-a-jax-rs-based -应用/

于 2015-01-16T14:58:56.263 回答
1

我找到了一种避免预期解决方案的方法。我只是使用字符串并使用 gson 库解析它们:

@POST
// @Consumes(MediaType.APPLICATION_JSON)
// @Produces(MediaType.APPLICATION_JSON)
public String getFilteredPictures(String jsonFilters)
{
    PictureProvider provider = new PictureProvider();
    ArrayList<PictureInfo> pictures = null;
    ArrayList<FilterOption> filters = null;
    if (jsonFilters != null)
    {
        Type collectionType = new TypeToken<ArrayList<FilterOption>>()
        {}.getType();
        filters = gson.fromJson(jsonFilters, collectionType);
    }
    .....
于 2013-10-11T12:57:28.627 回答
1

一个典型的原因MessageBodyProviderNotFoundException是您尝试序列化为 JSON 的类的格式不正确。

就我而言,我缺少一个无参数的构造函数。在我添加了一个空的无参数构造函数之后,一切都正常了。

于 2016-02-27T09:14:02.433 回答
0

FilterOption如果您告诉 JAX-RS 如何执行此操作,则JAX-RS 只能将 JSON 与实例相互转换。

一种常见的方法是为此使用 JAXB:

@XmlRootElement
class FilterOption {
    // members, getters, setters, constructors
}
于 2013-10-03T19:04:17.520 回答