Exit
, ExitAll
与ExitHook
Exit
: 仅退出当前执行的逻辑方法,不退出后续的请求流程,可用于替代return
。ExitAll
: 强行中断当前执行流程,当前执行方法的后续逻辑以及后续所有的逻辑方法将不再执行,常用于权限控制。ExitHook
: 当路由匹配到多个HOOK
方法时,默认是按照路由匹配优先级顺序执行HOOK
方法。当在HOOK
方法中调用ExitHook
方法后,后续的HOOK
方法将不会被继续执行,作用类似HOOK
方法覆盖。- 这三个退出函数仅在服务函数和
HOOK
事件回调函数中有效,无法控制中间件的执行流程。
由于ExitAll
和ExitHook
方法在应用层比较少用,因此这里仅介绍Exit
方法的使用。
Exit*
流程退出特性底层采用的是panic...recover...
机制来实现的,CPU执行损耗大约几十纳秒(ns
),通过极少的运行时开销来提高易用性。
Exit
返回方法
package main
import (
"github.com/gogf/gf/v2/frame/g"
"github.com/gogf/gf/v2/net/ghttp"
)
func main() {
s := g.Server()
s.BindHandler("/", func(r *ghttp.Request) {
if r.Get("type").Int() == 1 {
r.Response.Writeln("john")
}
r.Response.Writeln("smith")
})
s.SetPort(8199)
s.Run()
}
执行后,我们访问 http://127.0.0.1:8199/?type=1 ,可以看到页面输出了:
john
smith
我们将以上代码稍微调整一下:
package main
import (
"github.com/gogf/gf/v2/frame/g"
"github.com/gogf/gf/v2/net/ghttp"
)
func main() {
s := g.Server()
s.BindHandler("/", func(r *ghttp.Request) {
if r.Get("type").Int() == 1 {
r.Response.Writeln("john")
r.Exit()
}
r.Response.Writeln("smith")
})
s.SetPort(8199)
s.Run()
}
执行后,我们再次访问 http://127.0.0.1:8199/?type=1 ,可以看到页面输出了:
john
此外,Response
对象中提供了很多Write*Exit
的方法,表示输出内容后立即调用Exit
方法退出当前服务方法。