我正在弄乱Java中的HTTP和套接字,希望您能对此有所了解:
当我用Java SE 11编写的HTTP服务器没有读取整个请求然后进行响应时,客户端将无法获取该请求或发生错误。这是为什么?在服务器读取整个请求之前,客户端是否无法读取响应?如果在以下代码段中执行了对readBody的调用,则可以正常工作。如果响应具有Content-Length标头和文本正文,它也可以正常工作。这实际上让我感到困惑。
我的示例请求是带有数据 fds 的POST。邮递员说“无法获得任何请求”,而curl说“卷曲:(56)接收失败:对等重置连接”。
import java.io.*;
import java.net.Socket;
import java.util.*;
class Handler {
public synchronized void read(Socket incoming) {
try (incoming;
OutputStream outputStream = incoming.getOutputStream();
InputStream inputStream = incoming.getInputStream();
PrintWriter pw = new PrintWriter(outputStream)) {
writeRequest(inputStream);
pw.print("HTTP/1.1 200 OK\r\n");
pw.print("\r\n");
pw.flush();
} catch (IOException e) {
System.out.println("ERROR: " + e.getMessage());
e.printStackTrace();
}
}
private void writeRequest(InputStream inputStream) throws IOException {
String verbLine = readLine(inputStream);
Map<String, String> headers = readHeaders(inputStream);
//readBody(inputStream, headers);
}
private void readBody(InputStream inputStream, Map<String, String> headers) throws IOException {
Optional<String> optKey = headers.keySet().stream()
.filter(k -> k.equalsIgnoreCase("Content-Length"))
.findFirst();
if (optKey.isPresent()) {
int contentLength = Integer.parseInt(headers.get(optKey.get()));
byte[] bytes = inputStream.readNBytes(contentLength);
}
}
private Map<String, String> readHeaders(InputStream inputStream) throws IOException {
Map<String, String> headers = new HashMap<>();
while (true) {
String line = readLine(inputStream);
if (line == null || line.isEmpty()) {
return headers;
}
String key = line.split(":")[0].trim();
String value = line.split(":")[1].trim();
headers.put(key, value);
}
}
private String readLine(InputStream inputStream) throws IOException {
byte[] buf = new byte[200];
int offset = 0;
while (true) {
int read = inputStream.read();
if (read == -1) {
return null;
}
buf[offset] = (byte) read;
if (buf[0] == '\n' || (buf[0] == '\r' && buf[1] == '\n')) {
return "";
}
if (buf[offset] == 0x0A) {
int endOfLine = buf[offset - 1] == 0x0D ? offset - 1 : offset;
return new String(buf, 0, endOfLine);
} else {
offset++;
}
}
}
}
答案 0 :(得分:2)
如果在服务器上仍未读取数据的情况下关闭套接字,将导致客户端连接重置错误。因为您没有阅读完整的请求,所以在这里发生。如果尚未读取服务器的完整响应,则该错误将向用户显示。
如果您发送的响应是content-length
然后是完整的正文,则客户端将读取完整的响应,因此错误将被忽略。相反,如果您既不发送content-length
也未使用分块编码,则客户端将期望响应以适当的TCP连接关闭而结束。在这种情况下,由于尚未(正确)读取来自服务器的完整响应,因此连接重置将传播给用户。
答案 1 :(得分:0)
您的响应需要具有Content-Length标头或Transfer-Encoding标头-告诉客户端如何发送响应并允许其确定何时接收到所有字节。否则,它将需要等待EOF并假定响应主体被EOF终止(这是为了与HTTP / 1.0兼容)。您的客户可能不支持。 了解您正在使用哪个客户端可能会有所帮助。