Go中没有指针常量,const仅支持布尔、数字和字符串;指针不可变需靠变量作用域与接口设计实现,而非语法级const修饰。
直接说结论:const 在 Go 中不能声明指针类型的常量。Go 的 const 只支持布尔、数字(整型、浮点、复数)和字符串字面量,不支持指针、切片、map、struct 等复合类型,更不支持取地址操作(如 &x)出现在常量表达式中。
如果你看到类似 “指针常量” 的提法,大概率是混淆了以下概念:
int * const p(指针本身不可变,但指向内容可变)unsafe.Pointer 配合 uintptr 常量当作“常量指针”使用(实际是危险且非类型安全的)Go 没有语法级的“指针常量”,但可以通过设计让指针值在逻辑上不可变:
var globalConfigPtr = &defaultConfig),不提供 setterfunc Config() Reader),隐藏底层指针实现例如,模拟一个不可变配置引用:
type config struct {
Timeout int
}
var defaultConfig = config{Timeout: 30}
var configPtr = &defaultConfig // 包级变量,不导出赋值函数
func GetConfig() *config {
return configPtr // 调用方拿到指针,但无法修改 configPtr 本身
}
注意:configPtr 是变量,不是常量;它的值(即内存地址)可以被重新赋值,只是我们选择不这么做。
Go 语言本身不支持 const 修饰结构体字段或通过指针禁止写入(不像 C 的 const int* p)。即使你传入 *T,接收方仍可自由修改字段:
func inspect(c *config) {
c.Timeout = 999 // 合法!Go 不阻止这种写入
}若需逻辑上的“只读访问”,可行方案包括:
return *c),而非指针type ConfigReader interface { Timeou
t() int }),内部封装指针但不暴露字段Go 的设计哲学是“显式优于隐式”,它把“不可变性”交给程序结构和接口契约,而不是类型系统里的 const 修饰符。强行模拟会导致:
unsafe.Pointer 和 uintptr 常量,破坏内存安全与 GC 可达性func() *T 包裹指针并声称“这是常量”,实则每次调用都可能返回不同地址真正需要不可变数据时,优先考虑:是否该用值类型(struct)代替指针?是否该用 sync.Map 或不可变数据结构库(如 github.com/ericlagergren/decimal 的 immutability 设计)?
记住:Go 里最接近“指针常量”的安全实践,就是——不用指针,用值;不得不传指针时,用接口隔离可变性。