Go常见问题:command-line-argument: ***: undefined: ***

我们通常在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语言中,函数与方法不同)

run.go详情
在runRun()函数中,会发现files和cmdArgs接收的是传过来的文件列表,然后会通过GoFilesPackage(files),然后会入栈、加载、出栈等操作,由于启动的时候传递的只是一个.go文件,并没有传递demo.go,所以系统在加载main.go文件中并没有找到demo.go文件中定义的变量,则在p := load.GoFilesPackage(files)这一行,开始出错。

正确操作:

该出错原因属于go的多文件加载问题,采用go run命令执行的时候,需要把待加载的.go文件都包含到参数里面,正确操作如下图所示:

在这里插入图片描述

发布了255 篇原创文章 · 获赞 71 · 访问量 20万+

猜你喜欢

转载自blog.csdn.net/zhizhengguan/article/details/104267463