我正在使用Jersey(jaxrs-ri-2.7),Tomcat 7.0.47和Java 1.7.0_51。由于防火墙,我不能使用Maven。我的目标是做一个JSON RESTful Web服务。虽然有人告诉我MOXy包含在Jersey版本中,但我无法使用它。一方面,我可以让Jersey返回XML。当我通过浏览器调用Web服务时,得到的是:

org.glassfish.jersey.message.internal.MessageBodyProviderNotFoundException:

MessageBodyWriter not found for media type=application/json, type=class com.pracht.test.Calculation, genericType=class com.pracht.test.Calculation.

    at org.glassfish.jersey.message.internal.WriterInterceptorExecutor$TerminalWriterInterceptor.aroundWriteTo(WriterInterceptorExecutor.java:247)
    at org.glassfish.jersey.message.internal.WriterInterceptorExecutor.proceed(WriterInterceptorExecutor.java:162)
    at org.glassfish.jersey.server.internal.JsonWithPaddingInterceptor.aroundWriteTo(JsonWithPaddingInterceptor.java:103)
    at org.glassfish.jersey.message.internal.WriterInterceptorExecutor.proceed(WriterInterceptorExecutor.java:162)
    at org.glassfish.jersey.server.internal.MappableExceptionWrapperInterceptor.aroundWriteTo(MappableExceptionWrapperInterceptor.java:88)
    at org.glassfish.jersey.message.internal.WriterInterceptorExecutor.proceed(WriterInterceptorExecutor.java:162)
    at org.glassfish.jersey.message.internal.MessageBodyFactory.writeTo(MessageBodyFactory.java:1154)
    at org.glassfish.jersey.server.ServerRuntime$Responder.writeResponse(ServerRuntime.java:571)
    at org.glassfish.jersey.server.ServerRuntime$Responder.processResponse(ServerRuntime.java:378)
    ...


这是我的提供程序类:

package com.pracht.test;

import java.io.BufferedOutputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.lang.annotation.Annotation;
import java.lang.reflect.Type;

import javax.ws.rs.Produces;
import javax.ws.rs.WebApplicationException;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.MultivaluedMap;
import javax.ws.rs.ext.MessageBodyWriter;
import javax.ws.rs.ext.Provider;

import com.owlike.genson.Genson;
import com.owlike.genson.TransformationException;

@Provider
@Produces("application/json")
public class JsonProvider implements MessageBodyWriter<Calculation> {

    public JsonProvider() throws IOException{
        try {
            PrintWriter pw=new PrintWriter(
                       new BufferedOutputStream(new  FileOutputStream("c:\\test.txt")));
            pw.println("Reached JsonProvider()");
            pw.close();
        } catch(IOException e){
            e.printStackTrace();
        }
    }

    @Override
    public long getSize(Calculation arg0, Class<?> arg1, Type arg2,
            Annotation[] arg3, MediaType arg4) {
        return 0;
    }

    @Override
    public void writeTo(Calculation calculation,
            Class<?> type,
            Type genericType,
            Annotation[] annotations,
            MediaType mediaType,
            MultivaluedMap<String, Object> httpHeaders,
            OutputStream entityStream)
            throws IOException, WebApplicationException {
        Genson genson = new Genson();
        try {
            String json=genson.serialize(calculation);
            entityStream.write(json.getBytes());
        } catch (TransformationException e) {
            e.printStackTrace();
            throw new WebApplicationException("Serialization error",e);
        }
    }

    @Override
    public boolean isWriteable(Class<?> classType, Type arg1, Annotation[] arg2,
            MediaType arg3) {
        return classType == Calculation.class;
    }


}


我在构造函数中添加了PrintWriter,以希望告诉我至少可以构造一个对象并将结果放在可以看到的位置。我只是想快速了解一些内容,不想弄清楚正在使用什么日志记录API。

这是我的网络资源:

package com.pracht.test;

import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.PathParam;
import javax.ws.rs.Produces;
import javax.ws.rs.core.MediaType;

@Path("/calc")
public class CalcRest {


public CalcRest(){
    System.out.println("Reached CalcRest()");

}

@GET
@Path("/add/{a}/{b}")
@Produces("application/json")
public Calculation addPlainText(@PathParam("a") double a,
        @PathParam("b") double b) {
    System.out.println("Reached addPlainText()");
    return new Calculation().withProblem(a+"+"+b).withAnswer(a+b);
}

}


这是我使用Firefox拨打的电话:

http://localhost:8080/Contacts/rest/calc/add/1/2


附带说明一下,如果开箱即用的话,如果只有一个库实际上使用注释处理JSON Web服务,那么Java开发人员会喜欢吗?一个刚刚起作用?

最佳答案

您为什么为Genson编写MessageBodyWriter? Genson有它自己的类,当您在类路径中有Genson时,应自动选择该类。

如果不是这种情况,我猜泽西(Jersey)找到了另一个JSON提供程序(也许是moxy),因此不选择Genson。如果是这种情况,您可以尝试在类路径中创建一个文件夹META-INF / services并在其中定义Genson作为您的提供者(这应该覆盖其他提供者)。您可以在Genson here中查看它是如何完成的。

另外,您还应该使用最新版本的Genson 0.99,它有一些改进,例如JSR 353的实现,该实现试图为Java中的JSON解析(非数据绑定)提供“标准”。

关于java - 我如何告诉 Jersey 我的MessageBodyWriter在哪里?,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/23248121/

10-09 19:32