1 回答

TA贡献1820条经验 获得超9个赞
您可以使用 delve 调试运行时,但是您需要熟悉dlv
命令行界面(不使用 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命令反汇编 main func:
(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) --------- 运行直到断点或程序终止。
下一个(别名:n) ------------- 跳到下一个源代码行。
step (别名: s) ------------- 单步执行程序。
stepout (别名: so) --------- 跳出当前函数。
list (别名: ls | l) ------- 显示源代码。
args ----------------- 打印函数参数。
locals --------------- 打印局部变量。
如果我们将地图创建更改为abc := make(map[string]int, 100000)您将在反汇编中看到运行时函数已更改为runtime.makemap您可能更感兴趣的内容。
我希望这能让你继续前进,如果你需要澄清任何事情,请发表评论。
- 1 回答
- 0 关注
- 125 浏览
添加回答
举报