个人经验分享如何阅读Go语言源码
原文链接:分享如何阅读Go语言源码
前言
哈喽,大家好,我是Go源码包括哪些? 以我个人理解,asong
;最近在看Go语言调度器相关的源码,发现看源码真是个技术活,所以本文就简单总结一下该如何查看Go
源码,希望对你们有帮助。
Go
源码主要分为两部分,一部分是官方提供的标准库,一部分是Go
语言的底层实现,Go
语言的所有源码/标准库/编译器都在src
目录下:https://github.com/golang/go/...,想看什么库的源码任君选择;观看
Go
标准库 and Go
底层实现的源代码难易度也是不一样的,我们一般也可以先从标准库入手,挑选你感兴趣的模块,把它吃透,有了这个基础后,我们在看Go
语言底层实现的源代码会稍微轻松一些;下面就针对我个人的一点学习心得分享一下如何查看Go
源码;查看标准库源代码 标准库的源代码看起来稍容易些,因为标准库也属于上层应用,我们可以借助IDE的帮忙,其在IDE上就可以跳转到源代码包,我们只需要不断来回跳转查看各个函数实现做好笔记即可,因为一些源代码设计的比较复杂,大家在看时最好通过画图辅助一下,个人觉得画
UML
是最有助于理解的,能更清晰的理清各个实体的关系;有些时候只看代码是很难理解的,这时我们使用在线调试辅助我们理解,使用IDE提供的调试器或者
GDB
都可以达到目的,写一个简单的demo
,断点一打,单步调试走起来,比如你要查看fmt.Println
的源代码,开局一个小红点,然后就是点点点;文章图片
查看Go语言底层实现 人都是会对未知领域充满好奇,当使用一段时间
Go
语言后,就想更深入的搞明白一些事情,例如:Go程序的启动过程是怎样的,goroutine
是怎么调度的,map
是怎么实现的等等一些Go
底层的实现,这种直接依靠IDE跳转追溯代码是办不到的,这些都属于Go
语言的内部实现,大都在src
目录下的runtime
包内实现,其实现了垃圾回收,并发控制, 栈管理以及其他一些 Go 语言的关键特性,在编译Go
代码为机器代码时也会将其也编译进来,runtime
就是Go
程序执行时候使用的库,所以一些Go
底层原理都在这个包内,我们需要借助一些方式才能查看到Go
程序执行时的代码,这里分享两种方式:分析汇编代码、dlv调试;文章图片
分析汇编代码
前面我们已经介绍了
Go
语言实现了runtime
库,我们想看到一些Go
语言关键字特性对应runtime
里的那个函数,可以查看汇编代码,Go
语言的汇编使用的plan9
,与x86
汇编差别还是很大,很多朋友都不熟悉plan9
的汇编,但是要想看懂Go
源码还是要对plan9
汇编有一个基本的了解的,这里推荐曹大的文章:plan9 assembly 完全解析,会一点汇编我们就可以看源代码了,比如想在我们想看make
是怎么初始化slice
的,这时我们可以先写一个简单的demo
:// main.go
import "fmt"func main() {
s := make([]int, 10, 20)
fmt.Println(s)
}
有两种方式可以查看汇编代码:
1. go tool compile -S -N -l main.go
2. go build main.go && go tool objdump ./main
方式一是将源代码编译成
.o
文件,并输出汇编代码,方式二是反汇编,这里推荐使用方式一,执行方式一命令后,我们可以看到对应的汇编代码如下:文章图片
s := make([]int, 10, 20)
对应的源代码就是 runtime.makeslice(SB)
,这时候我们就去runtime
包下找makeslice
函数,不断追踪下去就可查看源码实现了,可在runtime/slice.go
中找到:文章图片
在线调试
虽然上面的方法可以帮助我们定位到源代码,但是后续的操作全靠
review
还是难于理解的,如果能在线调试跟踪代码可以更好助于我们理解,目前Go
语言支持GDB
、LLDB
、Delve
调试器,但只有Delve
是专门为Go
语言设计开发的调试工具,所以使用Delve
可以轻松调试Go
汇编程序,Delve
的入门文章有很多,这篇就不在介绍Delve
的详细使用方法,入门大家可以看曹大的文章:https://chai2010.cn/advanced-...,本文就使用一个小例子带大家来看一看dlv
如何调试Go
源码,大家都知道向一个nil
的切片追加元素,不会有任何问题,在源码中是怎么实现的呢?接下老我们使用dlv
调试跟踪一下,先写一个小demo
:import "fmt"func main() {
var s []int
s = append(s, 1)
fmt.Println(s)
}
进入命令行包目录,然后输入
dlv debug
进入调试$ dlv debug
Type 'help' for list of commands.
(dlv)
因为这里我们想看到
append
的内部实现,所以在append
那行加上断点,执行如下命令:(dlv) break main.go:7
Breakpoint 1 set at 0x10aba57 for main.main() ./main.go:7
执行
continue
命令,运行到断点处:(dlv) continue
> main.main() ./main.go:7 (hits goroutine(1):1 total:1) (PC: 0x10aba57)
2:
3: import "fmt"
4:
5: func main() {
6:var s []int
=>7:s = append(s, 1)
8:fmt.Println(s)
9: }
接下来我们执行
disassemble
反汇编命令查看main
函数对应的汇编代码:(dlv) disassemble
TEXT main.main(SB) /Users/go/src/asong.cloud/Golang_Dream/code_demo/src_code/main.go
main.go:50x10aba204c8d6424e8lea r12, ptr [rsp-0x18]
main.go:50x10aba254d3b6610cmp r12, qword ptr [r14+0x10]
main.go:50x10aba290f86f6000000jbe 0x10abb25
main.go:50x10aba2f4881ec98000000sub rsp, 0x98
main.go:50x10aba364889ac2490000000mov qword ptr [rsp+0x90], rbp
main.go:50x10aba3e488dac2490000000lea rbp, ptr [rsp+0x90]
main.go:60x10aba4648c744246000000000mov qword ptr [rsp+0x60], 0x0
main.go:60x10aba4f440f117c2468movups xmmword ptr [rsp+0x68], xmm15
main.go:70x10aba55eb00jmp 0x10aba57
=>main.go:70x10aba57*488d05a2740000lea rax, ptr [rip+0x74a2]
main.go:70x10aba5e31dbxor ebx, ebx
main.go:70x10aba6031c9xor ecx, ecx
main.go:70x10aba624889cfmov rdi, rcx
main.go:70x10aba65be01000000mov esi, 0x1
main.go:70x10aba6ae871c3f9ffcall $runtime.growslice
main.go:70x10aba6f488d5301lea rdx, ptr [rbx+0x1]
main.go:70x10aba73eb00jmp 0x10aba75
main.go:70x10aba7548c70001000000mov qword ptr [rax], 0x1
main.go:70x10aba7c4889442460mov qword ptr [rsp+0x60], rax
main.go:70x10aba814889542468mov qword ptr [rsp+0x68], rdx
main.go:70x10aba8648894c2470mov qword ptr [rsp+0x70], rcx
main.go:80x10aba8b440f117c2450movups xmmword ptr [rsp+0x50], xmm15
main.go:80x10aba91488d542450lea rdx, ptr [rsp+0x50]
main.go:80x10aba964889542448mov qword ptr [rsp+0x48], rdx
main.go:80x10aba9b488b442460mov rax, qword ptr [rsp+0x60]
main.go:80x10abaa0488b5c2468mov rbx, qword ptr [rsp+0x68]
main.go:80x10abaa5488b4c2470mov rcx, qword ptr [rsp+0x70]
main.go:80x10abaaae8f1dff5ffcall $runtime.convTslice
main.go:80x10abaaf4889442440mov qword ptr [rsp+0x40], rax
main.go:80x10abab4488b542448mov rdx, qword ptr [rsp+0x48]
main.go:80x10abab98402test byte ptr [rdx], al
main.go:80x10ababb488d35be640000lea rsi, ptr [rip+0x64be]
main.go:80x10abac2488932mov qword ptr [rdx], rsi
main.go:80x10abac5488d7a08lea rdi, ptr [rdx+0x8]
main.go:80x10abac9833d30540d0000cmp dword ptr [runtime.writeBarrier], 0x0
main.go:80x10abad07402jz 0x10abad4
main.go:80x10abad2eb06jmp 0x10abada
main.go:80x10abad448894208mov qword ptr [rdx+0x8], rax
main.go:80x10abad8eb08jmp 0x10abae2
main.go:80x10abadae8213ffbffcall $runtime.gcWriteBarrier
main.go:80x10abadf90nop
main.go:80x10abae0eb00jmp 0x10abae2
main.go:80x10abae2488b442448mov rax, qword ptr [rsp+0x48]
main.go:80x10abae78400test byte ptr [rax], al
main.go:80x10abae9eb00jmp 0x10abaeb
main.go:80x10abaeb4889442478mov qword ptr [rsp+0x78], rax
main.go:80x10abaf048c784248000000001000000mov qword ptr [rsp+0x80], 0x1
main.go:80x10abafc48c784248800000001000000mov qword ptr [rsp+0x88], 0x1
main.go:80x10abb08bb01000000mov ebx, 0x1
main.go:80x10abb0d4889d9mov rcx, rbx
main.go:80x10abb10e8aba8ffffcall $fmt.Println
main.go:90x10abb15488bac2490000000mov rbp, qword ptr [rsp+0x90]
main.go:90x10abb1d4881c498000000add rsp, 0x98
main.go:90x10abb24c3ret
main.go:50x10abb25e8f61efbffcall $runtime.morestack_noctxt
.:00x10abb2ae9f1feffffjmp $main.main
从以上内容我们看到调用了
runtime.growslice
方法,我们在这里加一个断点:(dlv) break runtime.growslice
Breakpoint 2 set at 0x1047dea for runtime.growslice() /usr/local/opt/go/libexec/src/runtime/slice.go:162
之后我们再次执行
continue
执行到该断点处:(dlv) continue
> runtime.growslice() /usr/local/opt/go/libexec/src/runtime/slice.go:162 (hits goroutine(1):1 total:1) (PC: 0x1047dea)
Warning: debugging optimized function
157: // NOT to the new requested capacity.
158: // This is for codegen convenience. The old slice's length is used immediately
159: // to calculate where to write new values during an append.
160: // TODO: When the old backend is gone, reconsider this decision.
161: // The SSA backend might prefer the new length or to return only ptr/cap and save stack space.
=> 162: func growslice(et *_type, old slice, cap int) slice {
163:if raceenabled {
164:callerpc := getcallerpc()
165:racereadrangepc(old.array, uintptr(old.len*int(et.size)), callerpc, funcPC(growslice))
166:}
167:if msanenabled {
之后就是不断的单步调试可以看出来切片的扩容策略;到这里大家也就明白了为啥向
nil
的切片追加数据不会有问题了,因为在容量不够时会调用growslice
函数进行扩容,具体扩容规则大家可以继续追踪,打脸网上那些瞎写的文章。上文我们介绍调试汇编的一个基本流程,下面在介绍两个我在看源代码时经常使用的命令;
- goroutines命令:通过
goroutines
命令(简写grs),我们可以查看所goroutine
,通过goroutine (alias: gr)
命令可以查看当前的gourtine
:
(dlv) grs
* Goroutine 1 - User: ./main.go:7 main.main (0x10aba6f) (thread 218565)
Goroutine 2 - User: /usr/local/opt/go/libexec/src/runtime/proc.go:367 runtime.gopark (0x1035232) [force gc (idle)]
Goroutine 3 - User: /usr/local/opt/go/libexec/src/runtime/proc.go:367 runtime.gopark (0x1035232) [GC sweep wait]
Goroutine 4 - User: /usr/local/opt/go/libexec/src/runtime/proc.go:367 runtime.gopark (0x1035232) [GC scavenge wait]
Goroutine 5 - User: /usr/local/opt/go/libexec/src/runtime/proc.go:367 runtime.gopark (0x1035232) [finalizer wait]
stack
命令:通过stack
命令(简写bt),我们可查看当前函数调用栈信息:
(dlv) bt
00x0000000001047e15 in runtime.growslice
at /usr/local/opt/go/libexec/src/runtime/slice.go:183
10x00000000010aba6f in main.main
at ./main.go:7
20x0000000001034e13 in runtime.main
at /usr/local/opt/go/libexec/src/runtime/proc.go:255
30x000000000105f9c1 in runtime.goexit
at /usr/local/opt/go/libexec/src/runtime/asm_amd64.s:1581
regs
命令:通过regs
命令可以查看全部的寄存器状态,可以通过单步执行来观察寄存器的变化:
(dlv) regs
Rip = 0x0000000001047e15
Rsp = 0x000000c00010de68
Rax = 0x00000000010b2f00
Rbx = 0x0000000000000000
Rcx = 0x0000000000000000
Rdx = 0x0000000000000008
Rsi = 0x0000000000000001
Rdi = 0x0000000000000000
Rbp = 0x000000c00010ded0
R8 = 0x0000000000000000
R9 = 0x0000000000000008
R10 = 0x0000000001088c40
R11 = 0x0000000000000246
R12 = 0x000000c00010df60
R13 = 0x0000000000000000
R14 = 0x000000c0000001a0
R15 = 0x00000000000000c8
Rflags = 0x0000000000000202[IF IOPL=0]
Cs = 0x000000000000002b
Fs = 0x0000000000000000
Gs = 0x0000000000000000
locals
命令:通过locals
命令,可以查看当前函数所有变量值:
(dlv) locals
newcap = 1
doublecap = 0
总结 看源代码的过程是没有捷径可走的,如果说有,那就是可以先看一些大佬输出的底层原理的文章,然后参照其文章一步步入门源码阅读,最终还是要自己去克服这个困难,本文介绍了我自己查看源码的一些方式,你是否有更简便的方式呢?欢迎评论区分享出来~。
【个人经验分享如何阅读Go语言源码】好啦,本文到这里就结束了,我是asong,我们下期见。
欢迎关注公众号:Golang梦工厂
推荐阅读
- 520专属Python代码分享
- 披荆斩棘成功上岸美团、字节、华为,分享Java面经及答案
- 查看个人过去的微信运动的运动数据的办法_微信
- 国庆节快乐的QQ留言代码分享_其它聊天
- 医院业务软件健康管理实战案例分享
- 适用于计算机和智能手机的十大个人防火墙
- Python小技巧练习分享
- Spring|Spring Boot+微信小程序_保存微信登录者的个人信息
- 开发Android应用程序的提示(我的经验教训)
- 原创工具14Finger-全能web指纹识别与分享平台