go语言包变量访问陷阱:init()函数中的局部变量遮蔽全局变量
本文分析一个Go语言包变量访问问题:utils包的init()函数中初始化的EsClient变量,在main函数中却为nil。
问题重现:
假设utils包代码如下:
package utils import ( "fmt" "log" "github.com/elastic/go-elasticsearch/v6" ) var EsClient *elasticsearch.Client func init() { // ... 配置省略 ... esClient, err := elasticsearch.NewClient(cfg) // 使用 := 声明局部变量 if err != nil { log.Fatal("连接失败", err) } fmt.Println(esClient) // 打印非nil值 }
main函数代码:
立即学习“go语言免费学习笔记(深入)”;
package main import ( "data_push/utils" "fmt" ) func main() { fmt.Println(utils.EsClient) // 打印 nil }
utils包的init()函数中,esClient变量使用:=声明,创建了一个新的局部变量,而非赋值给全局变量EsClient。因此,init()函数结束后,全局变量EsClient仍然为nil。
解决方案:
避免在init()函数中使用:=来声明全局变量。应使用=直接赋值:
package utils // ... 其他代码 ... func init() { // ... 其他代码 ... client, err := elasticsearch.NewClient(cfg) if err != nil { log.Fatal("连接失败", err) } EsClient = client // 使用 = 赋值全局变量 // ... 其他代码 ... }
修改后,main函数将能正确访问到已初始化的utils.EsClient。
总结:
Go语言的init()函数在包初始化时自动执行。 使用:=进行短变量声明时,需注意其作用域,避免局部变量遮蔽全局变量,导致意料之外的nil值。 使用=显式赋值全局变量可以避免此类问题。
以上就是Go语言包变量访问:为什么utils包的init函数中初始化的EsClient变量在main函数中却是nil?的详细内容,更多请关注php中文网其它相关文章!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号