在Servlet中发送和接收二进制数据

时间:2011-03-03 11:53:48

标签: http servlets post binary inputstream

我正在尝试使用HttpServletRequest.getOutputStream()HttpServletResponse.getInputStream()编写Java Servlet来接收二进制数据请求并回复它们。这适用于涉及通过HTTP POST连接向此Servlet响应的Silverlight客户端发送请求的项目。目前,为了测试Servlet,我正在用Java实现一个比Silverlight更熟悉的客户端。

问题在于,在我的测试项目中,我将来自Servlet客户端的数据作为字节数组发送,并期望接收具有相同长度的字节数组 - 只有它没有,而是我得到一个字节。因此,我在这里发布了相关的代码片段,希望您可以指出我在哪里做错了,并希望提供相关的参考书目来帮助我。

所以这里。

客户端 servlet处理来自非常简单的HTML页面的POST请求,其中包含我用作前端的表单。我并不太担心使用JSP等,而是专注于使Servlet间的通信工作

// client HttpServlet invokes this method from doPost(request,response)
private void process(HttpServletRequest request, HttpServletResponse response) 
throws ServletException, IOException {
  String firstName = (String) request.getParameter("firstname");
  String lastName = (String) request.getParameter("lastname");
  String xmlRequest = "<MyRequest><Person><Name Firstname=\""+firstName+"\" Lastname=\""+lastName+"\" /></Person></MyRequest>";

  OutputStream writer = null;
  InputStream reader = null;
  try {
    URL url = new URL("http://localhost:8080/project/Server");
    URLConnection conn = url.openConnection();
    conn.setDoInput(true);
    conn.setDoOutput(true);

    writer = conn.getOutputStream();
    byte[] baXml = xmlRequest.getBytes("UTF-8");
    writer.write(baXml, 0,baXml.length);
    writer.flush();
    // perhaps I should be waiting here? how?

    reader = conn.getInputStream();
    int available = reader.available();
    byte[] data = new byte[available];
    reader.read(data,0,available);
    String xmlResponse = new String(data,"UTF-8");

    PrintWriter print = response.getWriter();
    print.write("<html><body>Response:<br/><pre>");
    print.write(xmlResponse);
    print.write("</pre></body></html>");
    print.close();

  } finally {
    if(writer!=null)
      writer.close();
    if(reader!=null)
      reader.close();
  }
}

服务器 servlet处理HTTP POST请求。这是通过从上面的测试目的接收来自客户端Servlet的请求来完成的,但是将来我打算将它用于其他语言的客户端(特别是Silverlight)。

// server HttpServlet invokes this method from doPost(request,response)
private void process(HttpServletRequest request, HttpServetResponse response)
throws ServletException, IOException {
  ServletInputStream sis = null;
  try {
    sis = request.getInputStream();
    // maybe I should be using a BufferedInputStream 
    // instead of the InputStream directly?
    int available = sis.available();
    byte[] input = new byte[available];
    int readBytes = sis.read(input,0,available);
    if(readBytes!=available) {
      throw new ServletException("Oops! readBytes!=availableBytes");
    }

    // I ONLY GET 1 BYTE OF DATA !!! 
    // It's the first byte of the client message, a '<'.
    String msg = "Read "+readBytes+" bytes of "
                 +available+" available from request InputStream.";
    System.err.println("Server.process(HttpServletRequest,HttpServletResponse): "+msg);
    String xmlReply = "<Reply><Message>"+msg+"</Message></Reply>";
    byte[] data = xmlReply.getBytes("UTF-8");
    ServletOutputStream sos = response.getOutputStream();
    sos.write(data, 0,data.length);
    sos.flush();
    sos.close();

  } finally {
    if(sis!=null)
      sis.close();
  }
}

到目前为止,我一直坚持使用字节数组而不是使用BufferInputStream,因为我还没决定是否要使用例如用于传输数据的Base64编码字符串,或者我将按原样发送二进制数据。

提前谢谢你。

2 个答案:

答案 0 :(得分:8)

要将输入流复制到输出流,请使用标准方式:

InputStream is=request.getInputStream();
OutputStream os=response.getOutputStream();
byte[] buf = new byte[1000];
for (int nChunk = is.read(buf); nChunk!=-1; nChunk = is.read(buf))
{
    os.write(buf, 0, nChunk);
} 

答案 1 :(得分:3)

我能想到的一件事是你只读request.getInputStream().available()个字节,然后决定你已经拥有了所有东西。根据{{​​3}},available()将返回可以无阻塞地读取的字节数,但我没有看到是否实际上保证这是输入流的整个内容所以我倾向于认为没有这样的保证。

我不确定如何在没有更多数据的情况下找到最佳信息(请求中的Content-Length可以帮助吗?)而不会无意中阻止EOF阻塞,但我会尝试循环直到读完所有数据来自输入流的数据。为了测试该理论,您可以始终扫描输入以获得进一步发生在流中的已知模式,可能是>与您获得的初始<匹配。