登录
首页 >  Golang >  Go问答

查找gomap的初始化方法

来源:stackoverflow

时间:2024-03-28 21:42:27 145浏览 收藏

亲爱的编程学习爱好者,如果你点开了这篇文章,说明你对《查找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学习网!更多关于Golang的相关知识,也可关注golang学习网公众号。

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>