在编写http处理程序时,是否必须监听请求上下文的取消事件?
来源:stackoverflow
时间:2024-02-08 18:06:24 402浏览 收藏
欢迎各位小伙伴来到golang学习网,相聚于此都是缘哈哈哈!今天我给大家带来《在编写http处理程序时,是否必须监听请求上下文的取消事件?》,这篇文章主要讲到等等知识,如果你对Golang相关的知识非常感兴趣或者正在自学,都可以关注我,我会持续更新相关文章!当然,有什么建议也欢迎在评论留言提出!一起学习!
假设我正在编写一个 http 处理程序,在返回响应之前执行其他操作,我是否必须设置一个侦听器来检查 http 请求上下文是否已被取消?以便它可以立即返回,或者当请求上下文取消时是否有其他方法退出处理程序?
func handlesomething(w http.responsewriter, r *http.request) {
done := make(chan error)
go func() {
if err := dosomething(r.context()); err != nil {
done <- err
return
}
done <- nil
}()
select {
case <-r.context().done():
http.error(w, r.context().err().error(), http.statusinternalservererror)
return
case err := <-done:
if err != nil {
http.error(w, err.error(), http.statusinternalservererror)
return
}
w.writeheader(http.statusok)
w.write([]byte("ok"))
}
}
func dosomething(ctx context.context) error {
// simulate doing something for 1 second.
time.sleep(time.second)
return nil
}
我尝试对其进行测试,但是在上下文取消后,dosomething函数并没有停止并且仍在后台运行。
func TestHandler(t *testing.T) {
mux := http.NewServeMux()
mux.HandleFunc("/something", handleSomething)
srv := http.Server{
Addr: ":8989",
Handler: mux,
}
var wg sync.WaitGroup
wg.Add(1)
go func() {
defer wg.Done()
if err := srv.ListenAndServe(); err != nil {
log.Println(err)
}
}()
time.Sleep(time.Second)
req, err := http.NewRequest(http.MethodGet, "http://localhost:8989/something", nil)
if err != nil {
t.Fatal(err)
}
cl := http.Client{
Timeout: 3 * time.Second,
}
res, err := cl.Do(req)
if err != nil {
t.Logf("error: %s", err.Error())
} else {
t.Logf("request is done with status code %d", res.StatusCode)
}
go func() {
<-time.After(10 * time.Second)
shutdown, cancel := context.WithTimeout(context.Background(), 10*time.Second)
defer cancel()
srv.Shutdown(shutdown)
}()
wg.Wait()
}
func handleSomething(w http.ResponseWriter, r *http.Request) {
done := make(chan error)
go func() {
if err := doSomething(r.Context()); err != nil {
log.Println(err)
done <- err
}
done <- nil
}()
select {
case <-r.Context().Done():
log.Println("context is done!")
return
case err := <-done:
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
w.WriteHeader(http.StatusOK)
w.Write([]byte("ok"))
}
}
func doSomething(ctx context.Context) error {
return runInContext(ctx, func() {
log.Println("doing something")
defer log.Println("done doing something")
time.Sleep(10 * time.Second)
})
}
func runInContext(ctx context.Context, fn func()) error {
ch := make(chan struct{})
go func() {
defer close(ch)
fn()
}()
select {
case <-ctx.Done():
return ctx.Err()
case <-ch:
return nil
}
}正确答案
我刚刚对提供的解决方案进行了一些重构,现在它应该可以工作了。让我指导您完成相关更改。
dosomething 函数
func dosomething(ctx context.context) error {
fmt.printf("%v - dosomething: start\n", time.now())
select {
case <-ctx.done():
fmt.printf("%v - dosomething: cancelled\n", time.now())
return ctx.err()
case <-time.after(3 * time.second):
fmt.printf("%v - dosomething: processed\n", time.now())
return nil
}
}
它等待取消输入,或者在延迟 3 秒后返回给调用者。它接受要侦听的上下文。
handlesomething 函数
func handlesomething(w http.responsewriter, r *http.request) {
ctx := r.context()
fmt.printf("%v - handlerequestctx: start\n", time.now())
done := make(chan error)
go func() {
if err := dosomething(ctx); err != nil {
fmt.printf("%v - handlerequestctx: error %v\n", time.now(), err)
done <- err
}
done <- nil
}()
select {
case <-ctx.done():
fmt.printf("%v - handlerequestctx: cancelled\n", time.now())
return
case err := <-done:
if err != nil {
fmt.printf("%v - handlerequestctx: error: %v\n", time.now(), err)
w.writeheader(http.statusinternalservererror)
return
}
fmt.printf("%v - handlerequestctx: processed\n", time.now())
}
}
这里的逻辑和你的非常相似。在 select 中,我们检查接收到的错误是否为 nil ,并据此向调用者返回正确的 http 状态代码。如果我们收到取消输入,我们就会取消所有上下文链。
testhandler 函数
func TestHandler(t *testing.T) {
r := mux.NewRouter()
r.HandleFunc("/demo", handleSomething)
srv := http.Server{
Addr: ":8000",
Handler: r,
}
var wg sync.WaitGroup
wg.Add(1)
go func() {
defer wg.Done()
if err := srv.ListenAndServe(); err != nil {
fmt.Println(err.Error())
}
}()
ctx := context.Background()
ctx, cancel := context.WithTimeout(ctx, 1*time.Second) // request canceled
// ctx, cancel := context.WithTimeout(ctx, 5*time.Second) // request processed
defer cancel()
req, _ := http.NewRequestWithContext(ctx, http.MethodGet, "http://localhost:8000/demo", nil)
client := http.Client{}
res, err := client.Do(req)
if err != nil {
fmt.Println(err.Error())
} else {
fmt.Printf("res status code: %d\n", res.StatusCode)
}
srv.Shutdown(ctx)
wg.Wait()
}
在这里,我们启动一个 http 服务器并通过 http.client 向它发出 http 请求。可以看到有两条语句来设置上下文超时。如果您使用带有注释 // request canceled 的,则所有内容都会被取消,否则,如果您使用另一个,则请求将被处理。
我希望这能澄清您的问题!
文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《在编写http处理程序时,是否必须监听请求上下文的取消事件?》文章吧,也可关注golang学习网公众号了解相关技术文章。
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
478 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习