golang初探与命令源码分析

golang初探与命令源码分析,第1张

概述前段时间有群友在群里问一个go语言的问题: 就是有一个main.go的main函数里调用了另一个demo.go里的hello()函数。其中main.go和hello.go同属于main包。但是在main.go的目录下执行go run main.go却报hello函数没有定义的错: 代码结构如下: **gopath ---- src** **----gohello**

前段时间有群友在群里问一个go语言的问题:

就是有一个main.go的main函数里调用了另一个demo.go里的hello()函数。其中main.go和hello.go同属于main包。但是在main.go的目录下执行go run main.go却报hello函数没有定义的错:

代码结构如下:

**gopath ---- src**          **----gohello**                 **----hello.go**                     **?----main.go**

main.go如下:

package mainimport "fmt"func main() { fmt.Println("my name is main") hello()}

hello.go如下:

package mainimport "fmt"func hello() { fmt.Println("my name is hello")}

当时我看了以为是他GOPATH配置的有问题,然后自己也按照这样试了一下,报同样的错,在网上查了,也有两篇文章是关于这个错的,也提供了解决方法,即用go run main.go hello.go,试了确实是可以的。

虽然是个很简单的问题,但是也涉及到了go语言底层对于命令行参数的解析。那就来分析一下语言底层的实现吧,看一下底层做了什么,为什么报这个错?

分析:

以下使用到的Go SDK版本为1.8.3

该版本中go支持的基本命令有以下16个:

build       compile packages and dependencIEsclean       remove object filesdoc         show documentation for package or symbolenv         print Go environment informationBUG         start a BUG reportfix         run go tool fix on packagesfmt         run gofmt on package sourcesgenerate    generate Go files by processing sourceget         download and install packages and dependencIEsinstall     compile and install packages and dependencIEsList        List packagesrun         compile and run Go programtest        test packagestool        run specifIEd go toolversion     print Go versionvet         run go tool vet on packages

在Go SDK的src/cmd/go包下有main.go文件中,Command类型的commands数组对该16个命令提供了支持:

我们首先知道go语言的初始化流程如下:

在执行main.go中的主函数main之前,对import进来的包按顺序初始化,最后初始化main.go中的类型和变量,当初始化到commands数组时,由于cmdRun定义在于main.go同包下的run.go中,那么就先去初始化run.go中的变量和init方法,如下代码,先把cmdRun初始化为Command类型,然后执行init()函数。

var cmdRun = &Command{ Usageline: "run [build flags] [-exec xprog] gofiles... [arguments...]",Short:     "compile and run Go program",Long: `Run compiles and runs the main package comprising the named Go source files.A Go source file is defined to be a file ending in a literal ".go" suffix.By default,'go run' runs the compiled binary directly: 'a.out arguments...'.If the -exec flag is given,'go run' invokes the binary using xprog: 'xprog a.out arguments...'.If the -exec flag is not given,GOOS or GOARCH is different from the systemdefault,and a program named go_$GOOS_$GOARCH_exec can be foundon the current search path,'go run' invokes the binary using that program,for example 'go_nacl_386_exec a.out arguments...'. This allows execution ofcross-compiled programs when a simulator or other execution method isavailable.For more about build flags,see 'go help build'.See also: go build. `,}func init() { cmdRun.Run = runRun // break init loop addBuildFlags(cmdRun) cmdRun.Flag.Var((*stringsFlag)(&execCmd),"exec","")}

init()中,将runRun(其实类型是一个方法,用于处理run后的参数)赋值给cmdRu.run,addBuildFlags(cmdRun)主要是给run后面增加命令行参数(如:-x是打印其执行过程中用到的所有命令,同时执行它们)。其他15个命令和cmdRun类似,各有各的run实现。

下来主要看main.go中main的这块代码:

for _,cmd := range commands {   if cmd.name() == args[0] && cmd.Runnable() {     cmd.Flag.Usage = func() { cmd.Usage() }     if cmd.CustomFlags {       args = args[1:]     } else {       cmd.Flag.Parse(args[1:])       args = cmd.Flag.Args()     }     cmd.Run(cmd,args)     exit()     return   } }

这块代码遍历commands数组,当遍历到cmdRun时,cmd.name()其实就是拿到cmdRun.Usageline的第一个单词run

就会进入if分支,由于cmd.CustomFlags没有初始化故为false,走else分支,然后开始解析args命令行参数,args[1:]即取run后的所有参数。然后去执行cmd.Run(cmd,args),对于cmdRun来说,这里执行的就是run.go中init()的第一行赋值cmdRun.run(上面说了,这是一个函数,不同命令实现方式不同),即去执行run.go中的runRun函数,该函数主要是将命令行参数当文件去处理,如果是_test为后缀的,即测试文件,直接报错。如果是目录也直接报错(而且go run后面只能包含一个含main函数的go文件)。注意到有这么一行:

p := gofilesPackage(files)

gofilesPackage(files)除了校验文件类型和后缀,还会入栈,加载,出栈等 *** 作,由于启动的时候没有传递hello.go,所以系统加载main.go时找不到hello函数,导致报错。

本公众号免费提供csdn下载服务,海量IT学习资源,如果你准备入IT坑,励志成为优秀的程序猿,那么这些资源很适合你,包括但不限于java、go、python、springcloud、elk、嵌入式 、大数据、面试资料、前端 等资源。同时我们组建了一个技术交流群,里面有很多大佬,会不定时分享技术文章,如果你想来一起学习提高,可以公众号后台回复【2】,免费邀请加技术交流群互相学习提高,会不定期分享编程IT相关资源。

扫码关注,精彩内容第一时间推给你

总结

以上是内存溢出为你收集整理的golang初探与命令源码分析全部内容,希望文章能够帮你解决golang初探与命令源码分析所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/langs/1264740.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-06-08
下一篇 2022-06-08

发表评论

登录后才能评论

评论列表(0条)

保存