听取HTTP请求的正确方法是什么?

时间:2012-11-25 14:13:07

标签: java http webserver

我已经编写了一个非常简单的,相当低级别的HTTP(以及HTTP的一部分)服务器作为练习,以便更加熟悉我一直在避免的整个Web事物。它首次尝试的效果相当不错(并不是我推荐任何人实际使用它,但它会按照我的说法进行操作)。现在的问题是GET操作失败很多(刷新有帮助,但它不是很好 - 下面的详细信息),我认为这是因为我读取请求的方式(我相当确定我的路线有效):

void start()
{
    //...
    try(ServerSocket webSock = new ServerSocket(47000) {
    //...
        while(running) {
           try {
               Socket sock = webSock.accept();
               //read/write from/to sock 
           }
           //...
           Thread.sleep(10);
        }
    } 
}

(完整代码:http://pastebin.com/5B1ZuusH

我不确定 我做错了什么。

我收到错误:

This webpage is not available
The webpage at http://localhost:47000/ might be temporarily down or it may have moved permanently to a new web address.
Error 15 (net::ERR_SOCKET_NOT_CONNECTED): Unknown error.

相当多(整个页面无法加载),有时脚本或图像也不会加载。如果需要,我可以发布整个代码,但其余的主要是样板文件。

1 个答案:

答案 0 :(得分:3)

[另一次更新]

确定澄清我的回答,这是一个简单的Web服务器,它显示了如何读取GET请求。请注意,它在同一连接中处理多个请求。如果连接关闭,程序退出。通常,虽然我可以在连接关闭并退出程序之前从同一个Web浏览器发送一些请求。这意味着您不能使用end-of-stream作为消息结束的信号。

请注意,我从不使用手写的网络服务器来处理真实的事情。我最喜欢的是Tomcat,但其他框架也很好。

public class MyWebServer
{
   public static void main(String[] args) throws Exception
   {
      ServerSocket server = new ServerSocket(47000);
      Socket conn = server.accept();
      BufferedReader reader = new BufferedReader(new InputStreamReader(conn.getInputStream()));

      // don't use buffered writer because we need to write both "text" and "binary"
      OutputStream out = conn.getOutputStream();
      int count = 0;
      while (true)
      {
         count++;
         String line = reader.readLine();
         if (line == null)
         {
            System.out.println("Connection closed");
            break;
         }
         System.out.println("" + count + ": " + line);
         if (line.equals(""))
         {
            System.out.println("Writing response...");

            // need to construct response bytes first
            byte [] response = "<html><body>Hello World</body></html>".getBytes("ASCII");

            String statusLine = "HTTP/1.1 200 OK\r\n";
            out.write(statusLine.getBytes("ASCII"));

            String contentLength = "Content-Length: " + response.length + "\r\n";
            out.write(contentLength.getBytes("ASCII"));

            // signal end of headers
            out.write( "\r\n".getBytes("ASCII"));

            // write actual response and flush
            out.write(response);
            out.flush();
         }
      }
   }
}

[原始回复]

  

收听HTTP请求的正确方法是什么?

对于我们大多数人来说,正确的方法是使用设计良好的Web服务器框架,例如TomcatJettyNetty

  

作为练习更熟悉整个网络的事情

但是,如果这是学习HTTP的学术练习,那么首先要做的是研究HTTP协议(参见http://www.w3.org/Protocols/rfc2616/rfc2616.html)。我很确定你没有这样做,因为你的代码没有尝试识别起始行,标题等以确定GET请求何时完成并且发送响应是有意义的。

[更新]

冷却。您已经了解了TCP如何面向流并且不保留消息边界。是的,应用程序必须处理。这是最后一个想法 - 你可能会让你的实验相当可靠地工作 - 只有GET请求才会介意 - 如果你使用readLine来读取起始行和标题。当您得到一个空行时,请求就完成了。这将导致缓冲的阅读器在正确的时间阻止,以便您获得所有内容。

这不适用于POST等,因为您需要解析Content-Length标头并读取一定数量的字节。

希望当你意识到正确可靠地做到这一点时,这个实验会让你更加欣赏Jetty - 所以我觉得这是值得的。