亲爱的编程学习爱好者,如果你点开了这篇文章,说明你对《查找gomap的初始化方法》很感兴趣。本篇文章就来给大家详细解析一下,主要介绍一下,希望所有认真读完的童鞋们,都有实质性的提高。
问题内容我在vscode中使用delve调试galang代码。 我想使用delve来查看golang映射的创建,并且在使用make函数初始化映射时创建了一个断点,但它无法单步执行。
正确答案
您可以使用 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)
}
我们可以使用 dlv debug
命令让 delve 构建和调试我们的程序。这将使我们进入交互式调试器。
使映射调试起来“有趣”的原因之一是编译器可以以不同的方式创建映射。我们首先需要知道编译器使用哪些函数来创建映射(可以是多个函数,具体取决于映射大小)。我们将要求 delve 使用 disassemble -l main.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
,您可能对此更感兴趣。
我希望这能让您继续前进,如果您需要澄清任何事情,请发表评论。
以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持编程网!更多关于Golang的相关知识,也可关注编程网公众号。