卓越飞翔博客卓越飞翔博客

卓越飞翔 - 您值得收藏的技术分享站
技术文章34613本站已运行393

如果 golang http ResponseWriter 不超过 2kb,为什么会自动添加内容长度

如果 golang http responsewriter 不超过 2kb,为什么会自动添加内容长度

php小编新一将为您解答一个关于golang中的疑惑:为什么在处理http请求时,如果ResponseWriter的内容不超过2kb,还会自动添加内容长度?实际上,这是因为在http协议中,内容长度是一个必需的字段,它用于告知客户端要接收的数据长度,以便正确解析响应。即使内容长度较小,服务器仍然需要提供这个字段,以确保完整的通信过程。这样,客户端才能正确地接收并解析内容,保证请求和响应的完整性和准确性。

问题内容

func (handler Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
    var content string
    ...
    w.Write([]byte(content))
}

如果 len(content) <= 2048,则 content-length 将自动添加到响应中。而如果超过2048,则没有content-length,则会添加transfer-encoding: chunked

我找不到在哪里确定 2048。

我请求帮助查找在哪里确定 2048 的源代码。

解决方法

让我们看看http.responsewriter接口中该功能的文档,仅供参考清晰度:

[i]如果所有写入数据的总大小低于几 kb 并且没有 flush 调用,则会自动添加 content-length 标头。

首先,我们可以看到该数字可能不完全是 2048 (2 kb),但在我们预期的“几 kb”范围内。其次,我们可以看到此行为与 flush 方法有关,该方法记录在 flusher 中界面:

flush 将所有缓冲的数据发送到客户端。

flusher 接口由 responsewriter 实现,允许 http 处理程序将缓冲数据刷新到客户端。

默认的 http/1.x 和 http/2 responsewriter 实现支持 flusher,但 responsewriter 包装器可能不支持。处理程序应始终在运行时测试此功能。

正如它所说,您的 responsewriter 可能支持数据缓冲和刷新。这意味着当您将数据写入响应写入器时,它不会立即通过连接传输。相反,它首先被写入缓冲区。每次缓冲区太满而无法再写入时,当 servehttp 方法返回时,整个缓冲区将被传输。这可以确保即使您进行大量微小写入,数据也能有效传输,并且所有数据最终都能传输。您还可以选择使用 flush 方法随时主动清空缓冲区。 http 标头必须在正文数据之前发送,但在缓冲区第一次清空之前不需要发送它们。

把所有这些放在一起,你会发现如果写入的总量不超过缓冲区大小,并且我们从不调用 flush,那么在所有数据准备好之前不需要发送标头,此时点我们知道内容的长度。如果写入的总量大于缓冲区大小,则必须在知道内容长度之前发送标头,因此 responsewriter 无法自动确定。

这是在 net/http/server.go。具体来说,这里是缓冲区大小的声明,以及实现部分缓冲写入行为的 nchunkedwriter

// This should be >= 512 bytes for DetectContentType,
// but otherwise it's somewhat arbitrary.
const bufferBeforeChunkingSize = 2048

// chunkWriter writes to a response's conn buffer, and is the writer
// wrapped by the response.w buffered writer.
//
// chunkWriter also is responsible for finalizing the Header, including
// conditionally setting the Content-Type and setting a Content-Length
// in cases where the handler's final output is smaller than the buffer
// size. It also conditionally adds chunk headers, when in chunking mode.
//
// See the comment above (*response).Write for the entire write flow.
type chunkWriter struct {

1.19.5 的源代码链接。请注意,源代码可能会随着每个 go 版本的变化而变化。

卓越飞翔博客
上一篇: 均匀分布与天真的洗牌?
下一篇: 返回列表
留言与评论(共有 0 条评论)
   
验证码:
隐藏边栏