当前位置: 首页 > 面试题库 >

紧急堆栈跟踪中的未知字段

松旻
2023-03-14
问题内容

在尝试学习如何从恐慌中调试堆栈跟踪时,我遇到了一些令人困惑的问题。

package main

func F(a int) {
    panic(nil)
}

func main() {
    F(1)
}

当我在附加的播放链接上运行它时,输出以下内容:

panic: nil

goroutine 1 [running]:
main.F(0x1, 0x10436000)
    /tmp/sandbox090887108/main.go:4 +0x20
main.main()
    /tmp/sandbox090887108/main.go:8 +0x20

我无法解读第二个数字的含义(main.F(0x1,0x10436000)中的0x10436000)。如果有第二个int参数,或者作为第一个参数传递的其他内容,则不会出现(可以在第二个播放链接中看到)。

一个arg:https :
//play.golang.org/p/3iV48xlNFR

两个参数:https://play.golang.org/p/4jA7ueI86K


问题答案:

堆栈跟踪中打印的数据是参数,但是值并不直接与传入的参数相对应,它是原始数据,以指针大小的值打印(尽管通常这与本机字大小相同)。游乐场有点独特,因为它是一个带有32位指针(GOARCH=amd64p32)的64位字体系结构。

在traceback.go中,您可以看到通过根据指针大小逐步遍历参数来打印值。

for i := uintptr(0); i < frame.arglen/sys.PtrSize; i++ {

因此,由于单词大小是操场上指针大小的两倍,因此,在帧参数中始终会打印偶数个值。

通过在函数参数中使用较小的类型,可以在操场上看到如何显示数据的另一个示例: https

//play.golang.org/p/vHZDEHQZLh

func F(a uint8) {
panic(nil)
}

// F(1)
// main.F(0x97301, 0x10436000)

仅使用64位字的前8位,即0x97301 & 0x0f1。第0x97300一个值中的多余0x10436000部分和整个值只是该函数未使用的第一个64位字的其余部分。

amd64通过使用更多参数,您可以在系统上看到相同的行为。例如这个签名;

func F(a, b, c uint32)

通过调用时F(1, 1, 1),它将打印堆栈跟踪,例如:

main.F(0x100000001, 0xc400000001)

因为3个32位值需要2个字

最后要注意的一组值是返回值,它们也分配在堆栈上。以下功能签名:

func F(a int64) (int, int)

在amd64上,将显示如下堆栈框架参数:

main.F(0xa, 0x1054d60, 0xc420078058)

其中一个代表a,另一个代表(int, int)返回值。但是,返回值未初始化,因此,除了了解为什么存在这些值之外,在这里没有太多收获。



 类似资料:
  • 当Xdebug被激活时,只要PHP决定显示通知,警告,错误等,就会显示堆栈跟踪。堆栈跟踪显示的信息以及显示方式可以根据您的需要进行配置。 Xdebug在错误情况下显示的堆栈跟踪信息量相当保守(如果display.errors 在php.ini中设置为On)。这是因为大量的信息会减慢脚本的执行速度和浏览器中堆栈跟踪本身的渲染速度。但是,可以使堆栈轨迹以不同的设置显示更详细的信息。 堆栈跟踪中的变量

  • 我使用以下代码打印try-catch块中发生的任何异常,但是当异常发生时,logback不会打印完整的堆栈跟踪,而是写入一行错误(它没有明确说明是什么导致了它。我如何在logback输出中打印完整的堆栈跟踪? 尝试捕获异常的catch块 日志返回错误输出:

  • 问题内容: 有什么方法可以将堆栈跟踪作为字符串获取? 查看调试包(https://golang.org/pkg/runtime/debug/),它只能打印到标准输出。 问题答案: 将格式化的堆栈跟踪放到提供的中。然后,您可以将其转换为字符串。 您还可以使用,分配足够大的缓冲区来容纳整个堆栈跟踪,并使用将该跟踪放入其中,并返回缓冲区()。

  • 问题内容: 我编写了一个超简单的Java类,该类应按需抛出异常。但是我得到的堆栈跟踪看起来像这样: 注意:堆栈跟踪中没有行号,我希望有。 搜索此问题时找到的答案都是关于在编译时添加正确的参数,以确保行号确实进入了类文件。但是,我不认为这是我的问题,因为在我的ant build.xml中有此问题 而且,根据javap,看起来行号确实在: 那有什么呢?运行代码时,是否需要在jvm中设置参数? 谢谢!

  • 问题内容: 如何将异常的堆栈跟踪信息打印到stderr以外的流上?我发现的一种方法是使用getStackTrace()并将整个列表打印到流中。 问题答案: 可以接受or或参数: 也就是说,请考虑将SLF4J之类的记录器接口与LOGBack或log4j之类的记录实现一起使用。

  • 问题内容: 程序中没有单个方法“知道”它在堆栈中的位置。它所知道的只是它自己的小工作,它完成了并返回了。因此,当引发异常并打印堆栈跟踪时,它是从哪里来的? 在JVM中监视程序状态的每个应用程序旁边隐式地运行着一个单独的线程吗?还是JVM本身保存此信息,并且在抛出异常时以某种方式从异常中提取数据? 如果是上述两种情况之一,是否可以使用某些调用来检索堆栈跟踪(从监视器线程或JVM) 而不会 引发异常?