我们通常在go代码时,通常采用idea或者命令行的形式进行编译和运行,这两天我遇到一个非常困惑的问题,一直没找到具体原因,已经做了快一年的go程序员,看见这个问题甚是脸疼。
问题描述:
在开发代码过程中,经常会因为逻辑处理而对代码进行分类,放进不同的文件里面;像这样,同一个包下的两个文件,点击idea的运行按钮或者运行 go run main.go命令时,会报出如下错误,详情见图:
# command-line-arguments
src/demo/main/main.go:4:2: undefined: demo
Compilation finished with exit code 2
但是输入 go build,之后当前目录下会生成一个二进制文件,执行后会发现输出正确结果:
问题分析:
通过操作发现,输入go run main.go会执行失败,输入go build,在运行二进制文件可以成功,然后开始分析go源代码,找到执行命令的入口。
在源代码的main函数中,我们发现从base.Commands的切片中获取要执行的命令,然后和传入的args一起执行cmd.Run(cmd, args)这个方法;
然后再回过头看cmd.Run(cmd, args)这个函数,结果发现它只是定义了一种类型,具体实现这里没有指出;
紧接着回头去看run包下的函数,会发现run.go在初始化的时候,会把改文件下的runRun()函数赋值给base.Command{}定义的对象CmdRun,结果会发现runRun函数的的参数类型和个数完全符合cmd.Run(cmd, args)这个函数类型,在go语言中,函数的参数类型和参数个数符合定义的函数类型,则说明改函数实现了定义函数(注:go语言中,函数与方法不同)。
在runRun()函数中,会发现files和cmdArgs接收的是传过来的文件列表,然后会通过GoFilesPackage(files),然后会入栈、加载、出栈等操作,由于启动的时候传递的只是一个.go文件,并没有传递demo.go,所以系统在加载main.go文件中并没有找到demo.go文件中定义的变量,则在p := load.GoFilesPackage(files)这一行,开始出错。
正确操作:
该出错原因属于go的多文件加载问题,采用go run命令执行的时候,需要把待加载的.go文件都包含到参数里面,正确操作如下图所示: