如何查看go map的init

How to watch the init of the go map

我在 vscode 中使用 delve 调试 galang 代码。 我想用delve看golang map的创建,用make函数初始化map的时候打了个断点,就是进不去。

您可以使用 delve 来调试运行时,但是您需要熟悉 dlv command line interface(不使用 GUI)。

我将使用以下程序作为示例:

package main

func main() {
    abc := make(map[string]int)
    abc["a"] = 1
    abc["b"] = 2
    abc["c"] = 2
    fmt.Println(abc)
}

我们可以要求 delve 使用 dlv debug 命令构建和调试我们的程序。这将使我们进入交互式调试器。

地图调试“有趣”的原因之一是编译器可以用不同的方式创建地图。我们首先需要知道编译器使用哪些函数来创建地图(可以是几个,具体取决于地图大小)。我们将要求 delve 使用 disassemble -l main.main 命令反汇编主要功能:

(dlv) disassemble -l main.main
TEXT main.main(SB) /home/caveman/Downloads/test/main.go
        main.go:7       0x4b6860        64488b0c25f8ffffff              mov rcx, qword ptr fs:[0xfffffff8]
        main.go:7       0x4b6869        488d4424f0                      lea rax, ptr [rsp-0x10]
        main.go:7       0x4b686e        483b4110                        cmp rax, qword ptr [rcx+0x10]
        main.go:7       0x4b6872        0f8668010000                    jbe 0x4b69e0
        main.go:7       0x4b6878        4881ec90000000                  sub rsp, 0x90
        main.go:7       0x4b687f        4889ac2488000000                mov qword ptr [rsp+0x88], rbp
        main.go:7       0x4b6887        488dac2488000000                lea rbp, ptr [rsp+0x88]
        main.go:8       0x4b688f        e84c86f5ff                      call $runtime.makemap_small
        main.go:8       0x4b6894        488b0424                        mov rax, qword ptr [rsp]
        main.go:8       0x4b6898        4889442430                      mov qword ptr [rsp+0x30], rax
        ...

现在这看起来很复杂,但我们只需要查找对运行时的调用。在这种情况下 call $runtime.makemap_small 调用 makemap_small 函数。

既然我们知道了,我们就需要在这个运行时函数中打断点。我们使用 break mkSmallMap runtime.makemap_small 命令执行此操作:

(dlv) break mkSmallMap runtime.makemap_small
Breakpoint mkSmallMap set at 0x40eeef for runtime.makemap_small() /usr/local/go/src/runtime/map.go:292

现在我们可以通过执行c(继续)命令来启动我们的程序:

(dlv) c
> [mkSmallMap] runtime.makemap_small() /usr/local/go/src/runtime/map.go:292 (hits goroutine(1):1 total:1) (PC: 0x40eeef)
Warning: debugging optimized function
   287: }
   288:
   289: // makemap_small implements Go map creation for make(map[k]v) and
   290: // make(map[k]v, hint) when hint is known to be at most bucketCnt
   291: // at compile time and the map needs to be allocated on the heap.
=> 292: func makemap_small() *hmap {
   293:         h := new(hmap)
   294:         h.hash0 = fastrand()
   295:         return h
   296: }
   297:

我们中断了运行时函数本身。 help 命令将帮助您开始使用调试所需的所有命令。一些基本的:

  • 继续(别名:c)-------- 运行直到断点或程序终止。
  • next(别名:n)------------ 跳到下一个源代码行。
  • step(别名:s)------------ 单步执行程序。
  • stepout(别名:so)--------跳出当前函数。
  • list(别名:ls | l)------显示源代码。
  • args ---------------- 打印函数参数。
  • locals -------------- 打印局部变量。

如果我们将地图创建更改为 abc := make(map[string]int, 100000),您将在反汇编中看到运行时函数已更改为 runtime.makemap,您可能更感兴趣。

我希望这能让你继续下去,如果你需要澄清任何事情,请发表评论。