为什么生锈';s std::thread::sleep允许我的HTTP响应返回正确的正文?

为什么生锈';s std::thread::sleep允许我的HTTP响应返回正确的正文?,http,tcp,rust,Http,Tcp,Rust,我正在编写Rust编程语言最后一章的开头部分,该章将教授如何使用Rust编写HTTP响应 出于某种原因,正在发送的HTML文件不会显示在浏览器中,除非我在调用TcpResponse::flush()之前等待 代码如下: use std::io::prelude::*; use std::net::TcpListener; use std::net::TcpStream; use std::fs; use std::thread::sleep; use std::time::Duration;

我正在编写Rust编程语言最后一章的开头部分,该章将教授如何使用Rust编写HTTP响应

出于某种原因,正在发送的HTML文件不会显示在浏览器中,除非我在调用
TcpResponse::flush()
之前等待

代码如下:

use std::io::prelude::*;
use std::net::TcpListener;
use std::net::TcpStream;
use std::fs;
use std::thread::sleep;
use std::time::Duration;

fn main() {
    let listener = TcpListener::bind("127.0.0.1:7878").unwrap();

    for stream in listener.incoming() {
        let stream = stream.unwrap();

        handle_connection(stream);
    }
}

fn handle_connection(mut stream: TcpStream) {
    let mut buffer = [0; 1024];
    stream.read(&mut buffer).unwrap();

    let contents = fs::read_to_string("hello.html").unwrap();
    
    let response = format!(
        "HTTP/1.1 200 OK\r\nContent-Length: {}\r\n{}",
        contents.len(),
        contents
    );

    stream.write(response.as_bytes()).unwrap();
    // let i = stream.write(response.as_bytes()).unwrap();
    // println!("{} bytes written to the stream", i);
    // ^^ using this code instead will sometimes make it display properly
    sleep(Duration::from_secs(1));
    // ^^ uncommenting this will cause a blank page to load.

    stream.flush().unwrap();
}
我在多个浏览器中观察到相同的行为

根据铁锈书,调用
TcpListener::flush
应该确保字节完成对流的写入。那么,为什么我不能在浏览器中查看HTML文件,除非我在刷新之前休眠线程

我已经多次使用
cargo run
硬加载并重新启动服务器,行为也是一样的。我还将文件内容打印到终端,在任何一种情况下(当然是这样)都可以很好地读取内容

我想知道这是否是我的操作系统的问题。我在Windows10上

由于我可以继续学习,这并没有真正阻碍项目的进行(我现在也不打算将一个实际的web项目投入生产),但我希望任何人都能对这个问题有所了解。一定有一些关于Rust对溪流或环境的处理我不理解


谢谢你的时间

我看到您正在将传入的请求读入1024字节的缓冲区。您是否检查过该缓冲区是否足够大,可以读取整个请求?如果没有,请尝试扩大缓冲区。只是猜测而已。:)这不是有效的HTTP。
内容长度之后缺少一个
\r\n
:…\r\n
来分隔HTTP头和HTTP正文。请修复此问题并检查您的问题是否仍然存在。另一个问题是您没有任何
内容类型
标题,因此浏览器必须猜测您提供的内容类型。@PeterCharland在标题末尾需要有两个CRLF序列,一个在最后一个标题末尾,另一个用于标记标题结尾。换句话说,页眉和正文之间必须有一条空行。@PeterCharland:“这本书在告诉我…”-请仔细查看上的示例,您会发现缺少一个
\r\n
。从清单20-5中引用:
“HTTP/1.1200 OK\r\n内容长度:{}\r\n\r\n{}”
。这应该没有什么区别,但是您应该使用
流。write\u all
来确保整个响应都写入流。我看到您正在将传入的请求读取到1024字节的缓冲区中。您是否检查过该缓冲区是否足够大,可以读取整个请求?如果没有,请尝试扩大缓冲区。只是猜测而已。:)这不是有效的HTTP。
内容长度之后缺少一个
\r\n
:…\r\n
来分隔HTTP头和HTTP正文。请修复此问题并检查您的问题是否仍然存在。另一个问题是您没有任何
内容类型
标题,因此浏览器必须猜测您提供的内容类型。@PeterCharland在标题末尾需要有两个CRLF序列,一个在最后一个标题末尾,另一个用于标记标题结尾。换句话说,页眉和正文之间必须有一条空行。@PeterCharland:“这本书在告诉我…”-请仔细查看上的示例,您会发现缺少一个
\r\n
。从清单20-5中引用:
“HTTP/1.1200 OK\r\n内容长度:{}\r\n\r\n{}”
。这应该没有什么区别,但您应该使用
流。write\u all
来确保整个响应都写入流。