Exit, ExitAllExitHook

  1. Exit: 仅退出当前执行的逻辑方法,不退出后续的请求流程,可用于替代return
  2. ExitAll: 强行中断当前执行流程,当前执行方法的后续逻辑以及后续所有的逻辑方法将不再执行,常用于权限控制。
  3. ExitHook: 当路由匹配到多个HOOK方法时,默认是按照路由匹配优先级顺序执行HOOK方法。当在HOOK方法中调用ExitHook方法后,后续的HOOK方法将不会被继续执行,作用类似HOOK方法覆盖。
  4. 这三个退出函数仅在服务函数和HOOK事件回调函数中有效,无法控制中间件的执行流程。

由于ExitAllExitHook方法在应用层比较少用,因此这里仅介绍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方法退出当前服务方法。

Content Menu

  • No labels

4 Comments

  1. 2.0版

    r.GetInt("type") 要改成 r.Get("type").Int()
  2. 2.5.1新版本中间件调用了ExitAll方法,后置中间件好像还会执行下去,搞到有两个结果

    {
        "data": null,
        "t": "",
        "msg": "会话失效",
        "sub_msg": "请先登录,再发送请求",
        "code": 1001
    }{
        "data": null,
        "t": "504.9µs",
        "msg": "成功",
        "sub_msg": "成功",
        "code": 2000
    }

    1. ExitAll 后 ,后置中间件需要使用

      r.IsExited() 判断是否已经退出,这个不知道是就这样设计还是怎么回事,文档这块描述确实有点懵,这个问题也只会出现在response这类场景,因为前置中间件调response(如401鉴权提前输出)可能会有重复处理的问题。

      感觉这块文档可以再完善一下
      1. 再一个前置中间件可以通过seterr的方式统一让后置中间件处理,不提前输出,这个应当可以做个最佳实践