- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个设置了上下文截止日期的 Golang 程序。我正在发送 HTTP 请求,并希望在阅读正文时看到超出截止日期的错误。
似乎当我使用 ioutil.ReadAll
读取响应主体时,读取方法将被中断(?)并返回相应的错误(context.DeadlineExceeded
) .
但是,如果我使用 json.NewDecoder(resp.Body).Decode
读取响应正文,则返回的错误为 nil(而不是 context.DeadlineExceeded
)。我的完整代码如下。这是 json.NewDecoder(resp.Body).Decode
中的错误吗?
package main
import (
"context"
"encoding/json"
"fmt"
"io/ioutil"
"net/http"
"time"
)
var url string = "http://ip.jsontest.com/"
func main() {
readDoesntFail()
readFails()
}
type IpResponse struct {
Ip string
}
func readDoesntFail() {
ctx, _ := context.WithTimeout(context.Background(), time.Second*5)
req, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
if err != nil {
panic(err)
}
resp, err := http.DefaultClient.Do(req)
if err != nil {
panic(err)
}
ipResponse := new(IpResponse)
time.Sleep(time.Second * 6)
fmt.Println("before reading response body, context error is:", ctx.Err())
err = json.NewDecoder(resp.Body).Decode(ipResponse)
if err != nil {
panic(err)
}
fmt.Println("Expected panic but there was none")
}
func readFails() {
ctx, _ := context.WithTimeout(context.Background(), time.Second*5)
req, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
if err != nil {
panic(err)
}
resp, err := http.DefaultClient.Do(req)
if err != nil {
panic(err)
}
time.Sleep(time.Second * 6)
fmt.Println("before reading response body, context error is:", ctx.Err())
_, err = ioutil.ReadAll(resp.Body)
if err != nil {
fmt.Println("received expected error", err)
}
}
最佳答案
net/http
包可以使用缓冲区来处理请求。这意味着传入的响应正文可能会在您阅读之前部分或全部被读取和缓冲,因此过期的上下文可能不会阻止您完成阅读正文。而这正是发生的事情。
让我们修改您的示例以启动一个故意延迟响应(部分)的测试 HTTP 服务器:
ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
s := []byte(`{"ip":"12.34.56.78"}`)
w.Write(s[:10])
if f, ok := w.(http.Flusher); ok {
f.Flush()
}
time.Sleep(time.Second * 6)
w.Write(s[10:])
}))
defer ts.Close()
url = ts.URL
readDoesntFail()
readFails()
此测试服务器发送一个与 ip.jsontest.com
的响应类似的 JSON 对象。但它只发送 10 字节的主体,然后刷新它,然后在发送其余部分之前故意休眠 6 秒,“允许”客户端超时。
现在让我们看看调用 readDoesntFail()
会发生什么:
before reading response body, context error is: context deadline exceeded
panic: Get "http://127.0.0.1:38230": context deadline exceeded
goroutine 1 [running]:
main.readDoesntFail()
/tmp/sandbox721114198/prog.go:46 +0x2b4
main.main()
/tmp/sandbox721114198/prog.go:28 +0x93
在 Go Playground 上试试.
在您的示例中,json.Decoder.Decode()
读取已缓冲的数据,因此过期的上下文在这里不起作用。在我的示例中,json.Decoder.Decode()
尝试从连接中读取数据,因为数据还没有被缓冲(不可能是因为它还没有被发送),所以一旦上下文过期,从连接中进一步读取会返回超出截止日期的错误。
关于json - Go json.NewDecoder().Decode() 似乎不遵守上下文截止日期,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/72804689/
我正在使用 Skrollr 插件,它在桌面上运行良好。但在 iPad 上,我的文档的高度被 chop 了。我尝试删除所有 skrollr 数据属性,但问题仍然存在,我注意到只需激活插件就会遇到此问题。
我正在使用顶部栏作为我的顶部导航设置一个新站点。我的代码在空白页面上运行,所以我知道菜单结构是正确的,但是当我将它集成到我的网站设计中时,下拉菜单被截断/隐藏在主要内容后面。 未剪裁: 剪辑: 最佳答
这个问题已经有答案了: Unknown Column In Where Clause (16 个回答) 已关闭 5 年前。 当我在 MySQL 中尝试这个时: select make,count(*)
当 navController 出现在 UIPopoverController 中时,我的 UIBarButtonItem 在导航栏中被截断。知道是什么原因造成的吗? 问题截图:http://www.
我是一名优秀的程序员,十分优秀!