5

我编写了一个非常简单的、相当低级的 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.

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

4

1 回答 1

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();
         }
      }
   }
}

[来源回应]

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

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

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

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

[更新]

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

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

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

于 2012-11-25T14:32:08.670 回答