var App *Config App只是一个指针是吧?为什么可以直接使用config.App.Redis.DB这样呢?实体应该才会有字段值呀
这是一个非常棒的观察!这正是 Go 语言为了提升开发体验做的一个语法糖(Syntactic Sugar)。
在底层逻辑上,你的理解完全正确:指针本身确实只存储一个内存地址,真正的字段值存储在那个地址指向的物理空间(堆内存)里。
1. 自动解引用 (Automatic Dereferencing)
在很多老牌语言(如 C/C++)中,访问指针指向的结构体字段必须使用特定的符号,比如 app->Redis。
但在 Go 中,编译器非常聪明:当你写 config.App.Redis 时,如果它发现 App 是一个指针,它会自动帮你做解引用操作。
- 你写的:
config.App.Redis - 编译器实际执行的:
(*config.App).Redis
这让你既能享受指针带来的轻量传递(不用到处拷贝整个大结构体),又能享受像操作实体一样的简洁语法。
2. 为什么能拿到值?
当你执行 config.App = config.NewConfig() 时,发生了两件事:
- 分配内存:
&Config{...}在堆(Heap)上开辟了一块空间,填入了具体的配置值(Host, Port 等)。 - 存储地址:这个空间的"大门地址"被存进了
App这个指针变量里。
所以,当你通过 config.App.Redis.DB 访问时,程序是根据 App 里的地址找到了那块内存,并读取了对应偏移位置上的数据。
3. 一个致命的坑:空指针 (Nil Panic)
虽然语法上很方便,但指针有一个实体没有的风险:如果指针是 nil,程序会崩溃。
- 如果是实体:
var App Config。即便你不初始化,App.Redis.DB也会返回默认值0,不会报错。 - 如果是指针:
var App *Config。如果你忘记执行App = NewConfig()就直接访问App.Redis.DB,程序会直接报panic: runtime error: invalid memory address or nil pointer dereference。
总结
你之所以能直接用,是因为 Go 编译器帮你在底层偷偷加了一个 * 号。
避坑指南:
在使用 config.App.xxx 之前,一定要确保在 main 函数里已经完成了 config.App = LoadAll() 的赋值,否则程序启动就会报错。