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

Cub*_*bic 5 java webserver http

我编写了一个非常简单的、相当低级的 HTTP(嗯,HTTP 的一部分)服务器作为练习,以更加熟悉我一直避免的整个网络事物。它在第一次尝试时工作得相当好(我不会推荐任何人实际使用它,但它会按照我的指示去做)。现在的问题是 GET 操作失败了很多(刷新有帮助,但它不是很好 - 下面的详细信息),我认为这是因为我读取请求的方式(我相当确定我的路由有效):

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

(完整代码在这里: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.
Run Code Online (Sandbox Code Playgroud)

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

Gui*_*one 5

[另一个更新]

好的,澄清一下我的回答,这是一个简单的 Web 服务器,展示了如何读取 GET 请求。请注意,它在同一连接中处理多个请求。如果连接关闭,则程序退出。通常,尽管我可以在连接关闭和程序退出之前从同一个 Web 浏览器发送多个请求。这意味着您不能使用流结束作为消息结束的信号。

请注意,我从不使用手写的网络服务器来处理任何真实的事情。我最喜欢的是 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();
         }
      }
   }
}
Run Code Online (Sandbox Code Playgroud)

[原点响应]

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

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

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

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

[更新]

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

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

希望当您意识到正确和可靠地执行此操作涉及多少时,此实验将使您更加欣赏 Jetty - 所以我认为这是一项值得的努力。