【记录】go 设置 http 请求超时

背景

最近接手了一个老项目进行维护,发现其中有个关于 http 请求的方法设置的 timeout 没有生效,很奇怪!

一开始查看代码并没有发现什么可疑点,后查看了源码,打断点调试才发现问题所在,这里简单记录复盘一下。

说明 :本篇的源码的 go 版本是 1.20.2

问题

示例代码

go 复制代码
package main

import (
	"context"
	"fmt"
	"net/http"
	"time"
)

func main() {
	req, err := http.NewRequest(http.MethodGet, "https://www.baidu.com", nil)
	if err != nil {
		panic(err)
	}

	ctx, cancel := context.WithTimeout(context.Background(), time.Millisecond)
	defer cancel()
	req.WithContext(ctx)

	resp, err := http.DefaultClient.Do(req)
	if err != nil {
		panic(err)
	}

	//resp.Write(os.Stdout)
	fmt.Println("end: ", resp.StatusCode)
}

程序正常跑完并输出了,但是预期的是 http.DefaultClient.Do(req) 这里会直接报错,难道请求 1ms 就结束了??Why???

大家可以自己看下这段代码哪里有问题。

先说解决,其实就是 req.WithContext(ctx) 生成的是一个新的 http.Request 对象,上述的问题代码中并没有将其赋值给当前的 http.Request。大意了,没有闪。

复制代码
req = req.WithContext(ctx)

WithContext 方法的源码如下(net/http/request.go 356)

go 复制代码
func (r *Request) WithContext(ctx context.Context) *Request {
	if ctx == nil {
		panic("nil context")
	}
	r2 := new(Request)
	*r2 = *r
	r2.ctx = ctx
	return r2
}

请求超时设置

翻了下源码,看了下超时设置的方式,http 设置超时主要有两种方式:

  1. http.Client
go 复制代码
	c := http.Client{
		Timeout: time.Minute,
	}
	c.Do(req)
  1. http.Request 设置 context 超时
go 复制代码
	ctx, cancel := context.WithTimeout(context.Background(), time.Minute)
	defer cancel()
	req = req.WithContext(ctx)

Client 上指定 Timeout 会作用于通过该 Client 发起的所有请求,而 Request 设置 Context,仅针对这一次请求。使用的时候需要注意自己的场景。

设置 Tcp 连接阶段的超时可以这样:

复制代码
client := http.Client{
	Transport: &http.Transport{
		Dial: (&net.Dialer{
			Timeout:   2  * time.Second, //  tcp 连接时设置的连接超时
			Deadline:  time.Now().Add(3  * time.Second),  // 超时强制关闭
		}).Dial,
		TLSHandshakeTimeout: 2 * time.Second, //https 握手超时
	},
	Timeout: 5 * time.Second,
}

可以设置 Transport 中的 Dial

总结

平常自己使用 http 发送请求设置超时,都是直接给 http.Client 对象设置 Timeout 属性,很少使用这种对单个 Request 设置超时的。

其实还是个熟练度问题,平常源码读的比较少。有空读读源码不仅可以在使用代码的时候更得心应手,也能够学习借鉴源码的代码设计实现。对自己平常经常需要使用的库,还是建议都过一遍源码,很不错的一个打发空闲时间的方式。

相关推荐
pumpkin845147 小时前
01 Coze Studio项目中所有关键术语的详细解释和代码示例
golang
极安代理8 小时前
HTTP代理是什么?作用与场景全面解析
网络·网络协议·http
未来之窗软件服务9 小时前
幽冥大陆(一百07)—门禁局域网http获取名单—东方仙盟练气期
网络·http·仙盟创梦ide·东方仙盟·东方仙盟智能硬件·智能闸机
ps酷教程11 小时前
netty模拟文件列表http服务器
http·netty
AC赳赳老秦13 小时前
Dify工作流+DeepSeek:运维自动化闭环(数据采集→报告生成)
android·大数据·运维·数据库·人工智能·golang·deepseek
源代码•宸13 小时前
Leetcode—3. 无重复字符的最长子串【中等】
经验分享·后端·算法·leetcode·面试·golang·string
阳明Coding14 小时前
golang从入门到通天—数据库操作(gorm框架使用)(最简单最详细的golang学习笔记)
笔记·学习·golang
人道领域14 小时前
JavaWeb从入门到进阶(HTTP协议的请求与响应)
网络·网络协议·http
极安代理16 小时前
HTTP代理IP如何提升爬虫采集效率?
爬虫·tcp/ip·http
草根站起来17 小时前
https加密证书
网络协议·http·https