我正在编写一个 Java HTTP 服务器。我认为整个服务器都在工作并且它正在使用线程。但是,我意识到将请求读入 BufferedReader 的那段代码无法始终如一地工作。
这是读取传入请求的代码:
private String receive(WebSocket webSocket) throws IOException {
int chr;
System.out.println("Receiving!");
StringBuffer buffer = new StringBuffer();
while ( (chr = webSocket.in().read() ) != -1) {
buffer.append((char) chr);
if ( !webSocket.in().ready())
break;
}
return buffer.toString();
}
我的 Websocket 类只是包装了 Socket 并提供了输入和输出。我这样做是为了模拟套接字并测试我的服务器。
Websocket 类如下所示:
package http.server.socket;
import java.io.*;
import java.net.Socket;
public class SystemSocket implements WebSocket {
private Socket theConnection;
private BufferedReader in;
private OutputStream out;
public SystemSocket(Socket theConnection) throws IOException {
this.theConnection = theConnection;
in = new BufferedReader(new InputStreamReader(theConnection.getInputStream()));
out = new BufferedOutputStream(theConnection.getOutputStream());
}
public BufferedReader in() throws IOException {
return in;
}
public OutputStream out() throws IOException {
return out;
}
public void close() throws IOException {
in.close();
out.close();
theConnection.close();
}
}
问题是,对于用户在浏览器中输入的每个 url,都会发出两个请求 - 一个用于请求的页面,一个用于 favicon。有时 - 似乎 - 网站图标请求没有进入并且线程挂起。
这是我在一切顺利时打印到控制台的一些调试信息:
Receiving!
Receiving!
REQUEST STRING = GET /color_picker.html HT
[20130821 20:29:23] REQUEST: http://localhost:5000/color_picker.html
[20130821 20:29:23] PAGE RENDERED
REQUEST STRING = GET /favicon.ico HTTP/1.1
[20130821 20:29:23] REQUEST: http://localhost:5000/favicon.ico
[20130821 20:29:23] PAGE RENDERED
每当读取请求时,都会打印“正在接收”消息。所以,在这种情况下,“接收”消息被打印了两次,两个请求进来了,两个东西被渲染了。但是,同一个页面(但在不同的时间)会这样做(大约 10 秒后):
Receiving!
Receiving!
REQUEST STRING = GET /color_picker.html HTTP/1.1
[20130821 20:41:25] REQUEST: http://localhost:5000/color_picker.html
[20130821 20:41:25] PAGE RENDERED
REQUEST STRING =
Exception in thread "ServerThread" java.lang.ArrayIndexOutOfBoundsException: 1
at http.request.Parser.setRequestLineData(Parser.java:42)
at http.request.Parser.setRequestHash(Parser.java:27)
at http.request.Parser.parse(Parser.java:13)
at http.request.Request.get(Request.java:18)
at http.server.ServerThread.run(ServerThread.java:39)
所有后续错误都是因为请求字符串为空。但我无法弄清楚为什么请求字符串为空。我什至无法弄清楚如何调试。
有人可以帮忙吗??
还需要注意的是,如果第二个请求字符串没有立即进入,用户可以请求一个新的 url,这将导致第二个挂起的过程完成(所以第四个请求的 url 将是什么挂起)。因此,只有当用户停止请求时,在大约 10 秒后的最后一次请求中,我才会收到错误消息。有时我可以请求 20 个不同的页面,只有在我停止请求页面并等待几秒钟后,我才会看到错误。我想这是怎么回事??
更新:
根据请求,这里是 setRequestLineData() 方法:
private void setRequestLineData() {
requestHash = new HashMap<String, String>();
if (requestLineParts.length == 3) {
requestHash.put("httpMethod", requestLineParts[0]);
requestHash.put("url", requestLineParts[1]); //line 42
requestHash.put("httpProtocol", requestLineParts[2]);
}
else {
requestHash.put("httpMethod", requestLineParts[0]);
requestHash.put("url", requestLineParts[1]);
requestHash.put("queryString", requestLineParts[2]);
requestHash.put("httpProtocol", requestLineParts[3]);
}
}
更新:
我想我在导师的帮助下对这里发生的事情有了更多了解。他的想法是,一旦收到一个请求,浏览器会立即启动另一个请求,以减少下一个请求的加载时间。这听起来对我来说是合理的,因为我可以一页又一页地加载页面,但是在请求最后一页之后大约 10 秒我得到了一个错误。目前,我正在使用自定义异常处理此问题,但正在研究更好的解决方案。感谢所有的帮助家伙!