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
}
相关推荐
审判长烧鸡3 小时前
GO错误处理【2】os.Exit(1)/panic/返回err的应用场景
go·异常处理·panic
审判长烧鸡3 小时前
GO时区【4】PostgreSQL时区
postgresql·go
审判长烧鸡4 小时前
GO时区【3】字段与连接设置
postgresql·go
审判长烧鸡5 小时前
GO错误处理【1】不用try-catch用什么?
go·异常处理·try-catch·panic·fatal·os.exit
Go_error2 天前
Go database/sql 基于临时 channel 传递连接
后端·go
Go_error2 天前
Go 循环栅栏
后端·go
wecode662 天前
一个可以复现整个日志系统演进过程的工程级 specification
go·日志系统
程序设计实验室2 天前
当 CGO 遇见 Zig:一种更优雅的折腾方式,对比 GCC 后端
go
小熊吃保安3 天前
Excel下载变成了ZIP?Docker 容器里的 Content-Type 离奇失踪案
docker·go