登录
首页 >  Golang >  Go问答

golang - 如何在任何地方退出 http 处理程序

来源:stackoverflow

时间:2024-02-18 19:36:24 340浏览 收藏

学习Golang要努力,但是不要急!今天的这篇文章《golang - 如何在任何地方退出 http 处理程序》将会介绍到等等知识点,如果你想深入学习Golang,可以关注我!我会持续更新相关文章的,希望对大家都能有所帮助!

问题内容

我正在使用 net/http 包,想知道如何从代码中的任何位置退出处理程序。假设我有这个代码:

func main() {
    http.HandleFunc("/", handler)
    http.ListenAndServe(":8080", nil)
}

func handler(w http.ResponseWriter, r *http.Request){
    err := checkSomeThing(w, r)

    if err != nil {
        return
    }

    fmt.Println("End of Handler.")
    return
}

func checkSomeThing(w http.ResponseWriter, r *http.Request) error{
    http.Error(w, "Bad Request!", http.StatusBadRequest) 
    
    return errors.New("bad request")
}

理想情况下,我想从 checksomething 函数中退出处理程序,而不必返回,然后再次返回一个级别,随着应用程序的增长,情况会变得更糟。这纯粹是为了代码的可读性。


正确答案


惯用的方法是检查调用链上的错误返回。

要从任何地方退出处理程序,请按照 pattern in the encoding/json package 使用恐慌和恢复。

为恐慌定义一个独特的类型:

type httperror struct {
    status  int
    message string
}

编写一个要在 defer 语句中使用的函数。该函数检查类型并适当处理错误。否则,该函数将继续出现恐慌。

func handleexit(w http.responsewriter) {
    if r := recover(); r != nil {
        if he, ok := r.(httperror); ok {
            http.error(w, he.message, he.status)
        } else {
            panic(r)
        }
    }
}

编写一个辅助函数来调用panic:

func exit(status int, message string) {
    panic(httperror{status: status, message: message})
}

使用这样的函数:

func example() {
   exit(http.statusbadrequest, "bad!")
}

func somehandler(w http.responsewriter, r *http.request) {
   defer handleexit(w)
   example()
}

我的回答: 首先,golang 中建立的常见模式是让错误作为返回值从被调用者“冒泡”回调用者。它在可读性和可重用性方面具有很多优点。副作用是有很多 if err != nil {return} 检查。

如果你真的想打破常规,我的建议

我将提出一个想法,我认为它对于 golang 编码风格和模式来说并不常见或标准。但我在网上没有看到任何表明这是灾难性的信息。让我们看看我在评论中得到了什么说这太糟糕了。

您可以使用 runtime.Goexit() 来实现您想要的。处理程序只是等待另一个 goroutine 来完成工作。如果 go-routine 中运行的内部代码想要中止处理,可以调用 goexit()。它的优点是所有 defer 语句仍然会执行。

这似乎是 golang 目前不支持的异常处理的弱版本。但我把它扔在那里。

func handler(w http.ResponseWriter, r *http.Request) {

    var cleanExit bool = false
    var ch = make(chan bool)

    // the actual handler implementation in a goroutine
    go func() {
        defer close(ch)
        handlerImpl(w, r)
        cleanExit = true // if handlerImpl invokes goExit, this line doesn't execute
    }()

    // wait for goroutine to exit
    <-ch

    if cleanExit {
        fmt.Println("Handler exited normally")
    } else {
        fmt.Println("Hanlder was aborted")
    }
}

func handlerImpl(w http.ResponseWriter, r *http.Request) {
    checkSomeThing(w, r)
}

func checkSomeThing(w http.ResponseWriter, r *http.Request) {
    http.Error(w, "Bad Request!", http.StatusBadRequest)
    runtime.Goexit()
}

终于介绍完啦!小伙伴们,这篇关于《golang - 如何在任何地方退出 http 处理程序》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>