当前位置: 代码网 > it编程>前端脚本>Golang > Go中runtime.Caller的使用

Go中runtime.Caller的使用

2024年05月18日 Golang 我要评论
go中runtime.caller的使用下面介绍go runtime.caller使用runtime.caller函数定义func runtime.caller(skip int) (pc uintp

go中runtime.caller的使用

下面介绍go runtime.caller使用

runtime.caller
函数定义
func runtime.caller(skip int) (pc uintptr, file string, line int, ok bool)

作用
获取函数caller的调用信息

参数
skip=0时: 返回调用caller的函数a的pc(program counter)(a即runtime.caller代码当前所在函数)、所在文件名以及caller在函数a中的行数
skip=1时: 返回调用函数a的函数b的pc、所在文件名以及函数a在函数b中的行数;以此类推

pc: 程序计数器,指向下一条将要取指的指令地址
file: pc对应的函数所在文件路径
line: 被调用方在pc对应的函数中的行数

示例:
/users/go/src/liyouming/golang-trick/40-clean-code-controll-flow/util/util.go

package util
import (
	"bytes"
	"fmt"
	"runtime"
)
func callertest() {
	for i := 0; i <= 4; i++ {
		pc, file, line, ok := runtime.caller(i)
		if ok {
			fmt.printf("当i:=%d时:\n调用者的pc:%v\n调用者的函数名:%s\n"+
				"调用者所在file:%s\n被调用者在调用者中的line:%d\n", i, pc, runtime.funcforpc(pc).name(), file, line)
			fmt.println(string(bytes.repeat([]byte("*"), 10)))
		}
	}
}

main.go

package main
import (
	"golang-trick/40-clean-code-controll-flow/util"
)
func main() {
	util.callertest()
}

运行结果如下:

当i:=0时:
调用者的pc:17363077
调用者的函数名:golang-trick/40-clean-code-controll-flow/util.callertest
调用者所在file:/users/go/src/liyouming/golang-trick/40-clean-code-controll-flow/util/util.go
被调用者在调用者中的line:11
**********
当i:=1时:
调用者的pc:17363702
调用者的函数名:main.main
调用者所在file:/users/go/src/liyouming/golang-trick/40-clean-code-controll-flow/main.go
被调用者在调用者中的line:8
**********
当i:=2时:
调用者的pc:16987729
调用者的函数名:runtime.main
调用者所在file:/usr/local/opt/go/libexec/src/runtime/proc.go
被调用者在调用者中的line:250
**********
当i:=3时:
调用者的pc:17158656
调用者的函数名:runtime.goexit
调用者所在file:/usr/local/opt/go/libexec/src/runtime/asm_amd64.s
被调用者在调用者中的line:1594
**********

补充:

go学习——runtime.caller()函数

函数:

func caller(skip int) (pc uintptr, file string, line int, ok bool)
caller()报告当前go程调用栈所执行的函数的文件和行号信息。

参数解释:

  • skip

上溯的栈帧数,0表示caller的调用者(caller所在的调用栈)(0-当前函数,1-上一层函数,…)。

  • pc

调用栈标识符

  • file:

文件路径

  • line:

该调用在文件中的行号

  • ok:

如果无法获得信息,ok会被设为false

例子:

可能看了上面的解释,对于skip参数依然很迷惑,那我们来看个例子:

此时的项目目录结构:

blog/
	├── conf /...
	├── main.go
	├── middleware /...
	├── models /...
	├── pkg
	│ 	├── e /...
	│ 	├── logging
	│ 	│ 	├── file.go
	│ 	│ 	└── log.go
	│ 	├── setting /...
	│ 	└── util /...
	├── routers
	│ 	├── api
	│ 	│ 	├── auth.go
	│ 	│ 	└── v1
	│ 	│ 		├── article.go
	│ 	│ 		└── tag.go
	│ 	└── router.go
	├── runtime

就拿 blog/routers/api/v1/article.go当例子,在这个文件中getarticle()中用到了logging.info():

// followjianyustudygo/routers/api/v1/article.go
func getarticle(c *gin.context) {
	...
	} else {
		for _, err := range valid.errors {
			logging.info(err.key, err.message) // article.go:122  上溯栈帧数skip = 2
		}
	}
	...
}
// followjianyustudygo/pkg/logging/log.go
func info(v ...interface{}) {
	setprefix(info) // log.go:67 上溯栈帧数skip = 1
	logger.println(v)
}
func setprefix(level level) {
	_, file, line, ok := runtime.caller(defaultcallerdepth) // log.go:50 上溯栈帧数skip = 0
    ....
}

如果我们的skip

  • 为0:

代表上溯的栈帧数为0,返回的file就是调用caller()的位置:

[info][log.go:50]2022/04/29 21:07:11 [created_by 创建人不可以为空]
  • 为1:

代表上溯的栈帧数为1,返回的file就是调用caller()的上一层位置:

[info][log.go:67]2022/04/29 21:25:57 [state 状态只允许为0或1]
  • 为2:

代表上溯的栈帧数为2,返回的file就是调用caller()的上一层的上一层位置:

[info][article.go:122]2022/04/29 20:52:23 [state 状态只允许为0或1]

重点:
 因为我们在logging包里的log.go文件封装了info()、debug()、warn()、....函数,其他地方调用的都是这种封装好的函数,所以如果我们打印日志的时候,想要获取使用了logging.info()的位置,runtime.caller(skip int)的skip是要设置为2的(原因见skip=2的部分)!!!!

到此这篇关于go中runtime.caller的使用的文章就介绍到这了,更多相关go runtime.caller使用内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!

(0)

相关文章:

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论

验证码:
Copyright © 2017-2025  代码网 保留所有权利. 粤ICP备2024248653号
站长QQ:2386932994 | 联系邮箱:2386932994@qq.com