如何查看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
,您可能更感兴趣。
我希望这能让你继续下去,如果你需要澄清任何事情,请发表评论。
我在 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
,您可能更感兴趣。
我希望这能让你继续下去,如果你需要澄清任何事情,请发表评论。