GO结构体与指针【1】什么时候用指针

请问什么时候用指针什么时候用实体,例如多层级的配置信息,配置结构体载入做成接收者可以吗?

这是一个非常经典的问题。在 Go 中,配置信息(Config)通常建议使用指针或在初始化后作为全局只读变量。

1. 结构体字段建议:使用实体还是指针?

结论:结构体内部字段建议用实体,但对外传递和全局变量建议用指针。

  • 配置项(字段): 像 RedisConfig 这种小型结构体,直接用实体(Value)更好。它在内存中是连续的,对垃圾回收(GC)更友好。
  • 总配置(App): 建议将 var App Config 改为 var App *Config

2. 核心优化方案:修改 Load 模式

你目前的 Load 方法是在接收器上调用并返回新对象,这在逻辑上有点绕。更高效、更惯用的做法是直接初始化并返回指针。

修改 pkg\config\config.go

复制代码
package config

// 将全局变量设为指针
var App *Config

type Config struct {
	Server ServerConfig
	DB     map[string]DbConfig
	Redis  RedisConfig
	Logger LoggerConfig
}

// 改为函数,直接返回指针
func NewConfig() *Config {
	// 直接初始化,避免多次拷贝
	return &Config{
		Server: LoadServerConfig(), // 假设你改成了普通函数
		DB: map[string]DbConfig{
			"biz":    LoadDbConfig("biz"),
			"report": LoadDbConfig("report"),
		},
		Redis:  LoadRedisConfig(),
		Logger: LoadLoggerConfig(),
	}
}

修改 pkg\config\redis.go

复制代码
// 建议改为普通函数,返回实体值即可
func LoadRedisConfig() RedisConfig {
	return RedisConfig{
		Host:     getEnv("REDIS_HOST", "localhost"),
		Port:     GetEnvInt("REDIS_PORT", 6379),
		Password: getEnv("REDIS_PASSWORD", "123qwe"),
		DB:       GetEnvInt("REDIS_DB", 1),
	}
}

3. 为什么这样更高效?

  1. 减少拷贝: 你原来的 config.App = config.App.LoadAll() 会在栈上创建临时对象,然后再赋值给全局变量,产生两次内存拷贝。使用指针(*Config)后,全局变量只存储一个 8 字节的地址。
  2. 避免副作用: 你的 Load 方法使用了指针接收器 (r *RedisConfig) 但内部没用到 r,这容易产生误导。改为普通的 NewLoad 函数更清晰。
  3. 并发安全: 配置通常在 main 启动时一次性加载。一旦加载完成,全局指针指向的内存就不再变动,多协程并发读取 App.Redis.Host 是非常高效且安全的。

4. 改进后的 main 函数

复制代码
func main() {
	_ = godotenv.Load()
	
	// 初始化全局指针
	config.App = config.NewConfig()
    
	// 使用时很方便:config.App.Redis.Host
}
相关推荐
tyung1 天前
Go 手写 Wait-Free SPSC 无界队列:无 CAS、无锁、泛型节点池
数据结构·后端·go
踏着七彩祥云的小丑1 天前
Go学习第3天:变量+常量+运算符
开发语言·学习·golang·go
踏着七彩祥云的小丑2 天前
Go学习第2天:程序结构+基础语法+数据类型
开发语言·学习·golang·go
吴佳浩3 天前
AI Infra 的真相:Go 没输,rust也不是取代
后端·rust·go
2601_959644893 天前
2026年权威AI引擎优化服务咨询,专业之选
go
逐光老顽童3 天前
用 Go 实现一个 LLM 路由网关:Thompson Sampling 与自适应故障转移实践
vue.js·go
蓝宝石的傻话3 天前
MiBeeNvr v0.6.0: 延时摄影 + 转码界面 + ONVIF 增强 + 文档重构
go·github
先跑起来再说3 天前
Go 排行榜系统的工程化实现:分布式锁、快照表与定时刷新
分布式·go·gin
SenChien3 天前
Golang入门学习笔记
golang·go
唐青枫4 天前
别再把 make 和 new 搞混:Go make 从切片到通道实战详解
go