我們通常在go代碼時,通常采用idea或者命令行的形式進(jìn)行編譯和運行锡垄,這兩天我遇到一個非常困惑的問題,一直沒找到具體原因来破,已經(jīng)做了快一年的go程序員篮灼,看見這個問題甚是臉疼。
問題描述:
在開發(fā)代碼過程中徘禁,經(jīng)常會因為邏輯處理而對代碼進(jìn)行分類诅诱,放進(jìn)不同的文件里面;像這樣晌坤,同一個包下的兩個文件逢艘,點擊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它改,之后當(dāng)前目錄下會生成一個二進(jìn)制文件,執(zhí)行后會發(fā)現(xiàn)輸出正確結(jié)果:
問題分析:
通過操作發(fā)現(xiàn)商乎,輸入go run main.go會執(zhí)行失敗央拖,輸入go build,在運行二進(jìn)制文件可以成功鹉戚,然后開始分析go源代碼鲜戒,找到執(zhí)行命令的入口。
在源代碼的main函數(shù)中抹凳,我們發(fā)現(xiàn)從base.Commands的切片中獲取要執(zhí)行的命令遏餐,然后和傳入的args一起執(zhí)行cmd.Run(cmd, args)這個方法;
然后再回過頭看cmd.Run(cmd, args)這個函數(shù)赢底,結(jié)果發(fā)現(xiàn)它只是定義了一種類型失都,具體實現(xiàn)這里沒有指出;
緊接著回頭去看run包下的函數(shù)幸冻,會發(fā)現(xiàn)run.go在初始化的時候粹庞,會把改文件下的runRun()函數(shù)賦值給base.Command{}定義的對象CmdRun,結(jié)果會發(fā)現(xiàn)runRun函數(shù)的的參數(shù)類型和個數(shù)完全符合cmd.Run(cmd, args)這個函數(shù)類型洽损,在go語言中庞溜,函數(shù)的參數(shù)類型和參數(shù)個數(shù)符合定義的函數(shù)類型,則說明改函數(shù)實現(xiàn)了定義函數(shù)(注:go語言中碑定,函數(shù)與方法不同)流码。
在runRun()函數(shù)中,會發(fā)現(xiàn)files和cmdArgs接收的是傳過來的文件列表延刘,然后會通過GoFilesPackage(files)旅掂,然后會入棧、加載访娶、出棧等操作,由于啟動的時候傳遞的只是一個.go文件觉阅,并沒有傳遞demo.go崖疤,所以系統(tǒng)在加載main.go文件中并沒有找到demo.go文件中定義的變量秘车,則在p := load.GoFilesPackage(files)這一行,開始出錯劫哼。
正確操作:
該出錯原因?qū)儆?b>go的多文件加載問題叮趴,采用go run命令執(zhí)行的時候,需要把待加載的.go文件都包含到參數(shù)里面权烧,正確操作如下圖所示: