POST JSON与Jersey REST服务的问题

时间:2012-04-20 10:40:29

标签: java ajax json rest jersey

将JSON发布到Jersey REST服务时遇到问题 - GET工作正常但POST似乎很棘手。我一直在研究这个问题,到目前为止还没有解决方案。非常感谢任何帮助!

似乎无法找到U RL发送json?这是FireBug控制台显示的内容:

    POST http://localhost:9998/data 400 Bad Request
    Post source: name=Tony
    **Response Headers**
    Connection  close
    Content-Length  0
    Content-Type    text/html; charset=iso-8859-1
    Date    Fri, 20 Apr 2012 10:13:24 GMT
    **Request Headers**
    Accept  application/json, text/javascript, */*; q=0.01
    Accept-Encoding gzip, deflate
    Accept-Language sv-se,sv;q=0.8,en-us;q=0.5,en;q=0.3
    Connection  keep-alive
    Content-Length  9
    Content-Type    application/json; charset=UTF-8
    Host    localhost:9998
    Referer http://localhost:9998/static/page.html
    User-Agent  Mozilla/5.0 (Windows NT 6.1; WOW64; rv:11.0) Gecko/20100101 Firefox/11.0
    X-Requested-With    XMLHttpRequest

我按照以下方式进行POST:

<button id='btn' value="knapp" name="knapp" />
    <script type="text/javascript">
    $('#btn').click(function(){
        $.ajax({
            url: '/data',
            type: 'POST',
            contentType: 'application/json',
            data: {name:"Tony"},
            dataType: 'json'
        });
    })
</script>

带有@XmlRootElement的Javabean类:

@XmlRootElement
public class StatusBean {
    private String name;

    public StatusBean() {
    }

    public String getName() {
        return name;
    }

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

资源方法:

@Path("/data")
public class PostData {
    @POST
    @Consumes(MediaType.APPLICATION_JSON)
    @Produces(MediaType.APPLICATION_JSON)
    public StatusBean post(StatusBean sb) {
        System.out.println(sb);
        return sb;
    }
}

服务器,与Grizzly一起设置:

public class Main {
    public static final URI BASE_URI = getBaseURI();

    public static void main(String[] args) throws IOException {
        HttpServer httpServer = startServer();

        Map<String,String> initParams = new HashMap<String, String>();
        initParams.put("com.sun.jersey.config.property.packages", "server");
        SelectorThread selector = GrizzlyWebContainerFactory.create("http://localhost:9998/", initParams );

        System.out.println(String.format("Jersey app started with WADL available at "
                + "%sapplication.wadl\nTry out %shelloworld\nHit enter to stop it...",
                BASE_URI, BASE_URI));

        System.in.read();
        httpServer.stop();
    }

    protected static HttpServer startServer() throws IOException {
        System.out.println("Starting grizzly...");

        ClassNamesResourceConfig rc = new ClassNamesResourceConfig(PostData.class);

//        rc.getFeatures().put(JSONConfiguration.FEATURE_POJO_MAPPING, true);

        HttpServer server = GrizzlyServerFactory.createHttpServer(BASE_URI, rc);

        server.getServerConfiguration().addHttpHandler(new StaticHttpHandler(new File(".").getAbsolutePath()), "/static");

        return server;
    }

    private static int getPort(int defaultPort) {
        String port = System.getProperty("jersey.test.port");
        if (null != port) {
            try {
                return Integer.parseInt(port);
            } catch (NumberFormatException e) {
            }
        }
        return defaultPort;
    }

    private static URI getBaseURI() {
        return UriBuilder.fromUri("http://localhost/").port(getPort(9998)).build();
    }
}

5 个答案:

答案 0 :(得分:2)

  1. 尝试使您的bean可序列化。

    @XmlRootElement
    
    public class StatusBean implements Serializable {
    ....
    }
    
  2. 检查您的POST网址。应该是`

    http://localhost:9998/ {项目名称} / {restservletmapping} /数据

    例如,如果我的web.xml看起来像这样,我的项目名称是SampleProject

    <servlet-mapping>
    <servlet-name>Jersey REST Service</servlet-name>
    <url-pattern>/rest/*</url-pattern>
    </servlet-mapping>
    

    网址为:http://localhost:9998/SampleProject/rest/data

    您可以使用工具来测试REST服务,例如SOAP UI或浏览器插件,例如POSTMAN,REST CONSOLE等。

  3. 如果上面的内容很好,REST服务就会通过测试工具给出响应。 那么它可能是ajax中的跨源策略问题。

答案 1 :(得分:2)

我遇到了同样的问题。问题是您的数据未自动转换为JSON字符串。 所以你只需要在发布数据之前调用JSON.stringify(...):

<button id='btn' value="knapp" name="knapp" />
<script type="text/javascript">
$('#btn').click(function(){
    $.ajax({
        url: '/data',
        type: 'POST',
        contentType: 'application/json',
        data: JSON.stringify({name:"Tony"}),
        dataType: 'json'
    });
})
</script>

这应该有用。

答案 2 :(得分:1)

从您的服务器配置中我发现您尚未使用Grizzly配置JAX-RS。在example的基础上你应该以某种方式传递这样的属性

Map<String,String> initParams = new HashMap<String, String>();
initParams.put( "com.sun.jersey.config.property.packages", "package.with.your.StatusBean.class" );

另一种配置选项是使用

ResourceConfig rc = new PackagesResourceConfig("your.package.with.resources");

启动grizzly服务器:

GrizzlyServerFactory.createHttpServer(BASE_URI, rc);

查看详细信息:http://jersey.java.net/nonav/documentation/latest/user-guide.html(“部署根资源”一章)。尝试先运行他们的例子。

答案 3 :(得分:0)

您确定要发布的路径是否已完成?您应该在Path方法上定义另一个post注释,并在您发布到的网址中使用该注释:

@Path("/data")
public class PostData {
    @Path("/postStatus")
    @POST
    @Consumes(MediaType.APPLICATION_JSON)
    @Produces(MediaType.APPLICATION_JSON)
    public StatusBean post(StatusBean sb) {
        System.out.println(sb);
        return sb;
    }
}

然后使用/data/postStatus路径将您的请求发布到:

<button id='btn' value="knapp" name="knapp" />
<script type="text/javascript">
    $('#btn').click(function(){
        $.ajax({
            url: '/data/postStatus',

            type: 'POST',
            contentType: 'application/json',
            data: {name:"Tony"},
            dataType: 'json'
        });
    })
</script>

答案 4 :(得分:0)

您可能忘记注册JSON映射器,即Jackson(或您使用的任何映射器)。该功能未自动启用,您必须在ResourceConfig中加载该类:

strace /bin/echo

sample

also, see JSON howto