假设我有一个包文件夹,如下所示:
hello
|---main.go
|---utilities.go
在main.go
中,我有:
package main
在utilities.go
中,我有:
package hello
我这样做是因为:
main.go
文件中。 hello
文件夹中删除。但是当我运行go list hello
时,它给了我这个:
无法加载包:包hello:找到包main(main.go)和 hello(utilities.go)在E:\ Workbench \ Go \ src \ hello
中
然后我删除utilities.go
并仅使用go list hello
再次尝试main.go
。它给了我这个:
您好
所以我必须将所有实用程序移动到另一个包文件夹吗?
我是否必须在hello包文件夹中只保留单个main.go
,因为它似乎与其他包名称不相符?
有用的参考:http://thenewstack.io/understanding-golang-packages/
经过几次简短的实验后,我猜import yyy
决定了链接搜索路径,而package xxx
声明决定了已编译的.a
文件中的导出符号名称。
因为我注意到了这些事实:
package xxx
声明不必与包含的包文件夹名称相同,例如yyy
。import yyy
声明必须使用包文件夹路径yyy
。xxx.Symbol
而不是yyy.Symbol
。为了证实我在ADD 2中的猜测,我做到了这一点:
(以pkg1 \ file1.go)
package pkg2 // I deliberately use a different pkg name "pkg2" rather than the folder name "pkg1"
import "fmt"
func FA() {
fmt.Println("in the pkg2.FA")
}
(CMD1 \ main.go)
package main
import "pkg1"
func main() {
pkg2.FA() //here, the FA is referenced via "pkg2" rather than the "pkg1".
}
以上2个文件可以编译运行。
但如果我改变了这个:
(CMD1 \ main.go)
package main
import "pkg1"
func main() {
pkg1.FA() //here, the FA is referenced via "pkg1" as in the import statement. Should work, isn't it?
}
这将给出:
.... \ src \ cmd1 \ main.go:3:已导入但未使用:“pkg1”为pkg2
.... \ src \ cmd1 \ main.go:6:undefined:pkg1中的pkg1.FA
要进行双重确认,我使用pkg1.a
检查了已编译的go tool nm pkg1.a
文件,我没有看到包名影响导出的符号。 (顺便说一句,GNU nm工具似乎无法识别golang .a
文件格式。)
U
515 T %22%22.FA <========== this is my fuction FA
67b R %22%22.FA·f <========== this is my fuction FA
5eb T %22%22.init
67b B %22%22.initdone·
683 R %22%22.init·f
U fmt.Println
U fmt.init
673 R gclocals·33cdeccccebe80329f1fdbee7f5874cb
66b R gclocals·69c1753bd5f81501d95132d08af04464
65b R gclocals·e29b39dba2f7b47ee8f21f123fdd2633
64d R go.string."in the pkg2.FA"
63d R go.string.hdr."in the pkg2.FA"
7d2 R go.typelink.*[1]interface {}
796 R go.typelink.[1]interface {}
737 R go.typelink.[]interface {}
U runtime.algarray
U runtime.convT2E
6ec R runtime.gcbits.01
68b R runtime.gcbits.03
U runtime.morestack_noctxt
U runtime.throwinit
79a R type.*[1]interface {}
7d6 R type..importpath.fmt.
73b R type..namedata.*[1]interface {}.
6ed R type..namedata.*[]interface {}.
68c R type..namedata.*interface {}.
74e R type.[1]interface {}
6ff R type.[]interface {}
69c R type.interface {}
U type.string
所以到目前为止的结论是:
当声明的包名称与包文件夹名称不同时,包声明将用作包的备用名称。
相当于此,但隐含地:
import pkg2 "pkg1"
顺便说一下,由于包声明没有反映在编译的.a
文件中,我想golang编译需要.go
文件的存在。
答案 0 :(得分:3)
将文件放在同一个包中AKA文件夹必须具有相同的包名。唯一的例外是可以具有_test扩展的测试。该规则是1个文件夹,1个包因此是同一文件夹中所有go文件的相同包名。
答案是肯定的,将“hello”软件包文件移到他们自己的文件夹中,或者对同一目录中的所有文件使用相同的软件包名称。