golang 包导入流程如何工作?是否有一种方法可以在同一模块中多次初始化一个包?

golang 包导入流程如何工作?是否有一种方法可以在同一模块中多次初始化一个包?

问题内容

我是 golang 新手,目前正在研究包和导入它们的工作原理。我对这次进口有几个问题。我正在阅读这篇文章(它有超过 7k 个赞,所以我猜它是准确的 + 这是当我输入 golang 包 时 google 给我的第一个内容)。

为了解释我不明白的地方,我先写一下项目结构是什么样的,所有文件的内容是什么。

pkg├── mypkg|  └── mypkg.go|├── pkg1|  └── pkg1.go|└── pkg2   └── pkg2.gogo.modmain.go

登录后复制

如您所见,我有包含 3 个包的模块(4 个包包括主包)。下面列出了所有包的内容。

pkg1.go 的内容

立即学习“go语言免费学习笔记(深入)”;

package pkg1import (    "fmt"    "mymod/pkg/mypkg")func init() {    fmt.println("hello everyone from pkg1 init")}func hellofrompkg1() {    fmt.println("hello from pk1")    mypkg.print()}

登录后复制

pkg2.go 的内容

package pkg2import (    "fmt"    "mymod/pkg/mypkg")func init() {    fmt.println("hello everyone from pkg2 init")}func hellofrompkg2() {    fmt.println("hello from pk2")    mypkg.print()}

登录后复制

mypkg 的内容

package mypkgimport "fmt"func init() {    fmt.println("hello everyone from mypkg init")}var prom = 10func print() {    fmt.printf("address of prom inside mypkg is: %p", &prom)}

登录后复制

ma​​in.go 的内容

package mainimport (    "fmt"    "mymod/pkg/pkg1"    "mymod/pkg/pkg2")func init() {    fmt.println("hello everyone from main init")}func main() {    pkg1.hellofrompkg1()    pkg2.hellofrompkg2()}

登录后复制

因此,main.go 包含 pkg1 和 pkg2,并且 pkg1 和 pkg2 都包含 mypkg。我引用的文章指出了以下内容(以粗体显示):

the main thing to remember is, an imported package is initialized only once per package.

登录后复制

考虑到这一点,我期望我的程序的输出是这样的:

hello everyone from mypkg inithello everyone from pkg1 inithello everyone from mypkg inithello everyone from pkg2 inithello everyone from main inithello from pk1address of prom inside mypkg is: 0xfee360 (some address)hello from pk2address of prom inside mypkg is: 0xf321a3 (another address)

登录后复制

我的期望是遵循以下步骤:

进入主包(1)pkg1 包已初始化 (2)mypkg 包已初始化 (2.1)mypkg 中的所有全局变量均已初始化 -prom 在我的情况下 (2.1.1)调用 mypkg 的 init 函数(2.1.2)调用 pkg1 的 init 函数(2.2)pkg2 包已初始化 (3)mypkg 包已初始化 (3.1)mypkg 中的所有全局变量均已初始化 -prom 在我的情况下 (3.1.1)调用 mypkg 的 init 函数(3.1.2)调用pkg2的init函数(3.2)主包初始化(4)调用main的init函数(5)主包的main函数被调用(6)

取而代之的是,我得到以下输出:

Hello everyone from mypkg initHello everyone from pkg1 initHello everyone from pkg2 initHello everyone from main initHello from pk1address of prom inside mypkg is: 0x8fe360Hello from pk2address of prom inside mypkg is: 0x8fe360

登录后复制

看起来 mypkg 在第一次导入期间只初始化了一次?!另外,全局变量 prom 在 pkg1 和 pkg2 中的地址是相同的(在本例中为 0x8fe360)。

所以我的问题是:

文章作者是否犯了错误?导入的包不是每个包仅初始化一次,而是每个模块一次?这是否意味着一个包中的全局变量在整个模块中始终是相同的(相同的地址),无论该包被包含多少次以及从何处包含?我的意思是它们只会在首次导入期间初始化一次?有什么方法可以让我的“心流”发挥作用吗?我的意思是每次导入都会独立初始化一个包?在我的示例中,这意味着 mypkg 在 pkg1 中初始化一次,在 pkg2 中初始化另一次。如果我读的内容不正确,是否有人有一篇关于 go 中的包和模块的好文章?

我知道对于某些人来说这些问题是关于 golang 的一些基础知识,但对于我作为初学者来说这引起了一些误解。更重要的是,该程序的工作结果与谷歌搜索中首先出现的文章的作者所写的内容不符。欢迎任何帮助。祝一切顺利!

正确答案

程序中包含的包是一个集合,它是从 main 开始的所有导入包的传递闭包。即:

这是一套。每个导入的包仅包含一次。这意味着,如果您在包中定义了一个变量,则该变量仅出现一次。所有导入的包以及它们递归导入的包都包含在最终的二进制文件中。

至于初始化:您的步骤是正确的,只是 mypkg 只初始化一次。二进制文件中没有 mypkg 的多个副本。

以上就是golang 包导入流程如何工作?是否有一种方法可以在同一模块中多次初始化一个包?的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2485551.html

(0)
上一篇 2025年3月4日 21:47:45
下一篇 2025年3月3日 10:17:35

AD推荐 黄金广告位招租... 更多推荐

相关推荐

发表回复

登录后才能评论