Go通过标识符首字母大小写控制导出:大写字母开头的标识符(如GetData、Config、MaxRetries)可被其他包访问,小写开头的(如processData、internalHelper、defaultTimeout)仅包内可见;结构体字段和匿名字段同样遵循此规则,且需正确导入包路径才能使用。
在 Go 语言中,没有像其他语言那样的 public、private 关键字,而是通过**标识符首字母大小写**来控制导出(即对外可见)与否。这是 Go 包设计的核心规则之一,直接影响函数、变量、类型、常量等能否被其他包访问。
Go 规定:只有首字母为**大写**(Unicode 大写字母,如 A–Z)的标识符才被视为“导出的”,可在包外被引用;小写字母开头的标识符(如 myFunc、count)仅在当前包内可见。
func GetData() string → 可被其他包调用(导出函数)func processData() error → 仅本包可用(未导出,相当于“私有”)type Config struct{...} → 可被外部使用type internalHelper struct{...} → 无法从包外实例化或嵌入const MaxRetries = 3 → 导出常量const defaultTimeout = 5000 → 仅包内可见变量本身是否导出,取决于其名称;但结构体字段是否可被外部访问,还取决于字段名是否大写——即使该结构体已导出。
type User struct { Name string; age int }Name 可被外部读写,age 在包外不可见(哪怕 User 是导出类型)func (u *User) Age() int { return u.age }
一个标识符即使导出,也必须通过正确的包导入路径才能使用。Go 不支持跨目录隐式引用,也不允许循环导入。
github.com/you/app/utils,其中定义了 func ValidateEmail(...)
import "github.com/you/app/utils",再调用 utils.ValidateEmail(...)
utils 但位于非标准路径(如本地 ./internal/utils),则无法被外部模块导入(internal 目录下包对模块外不可见)新手容易混淆“能编译”和“能导出”,
或误以为加注释、加标签可改变可见性——实际上 Go 的导出机制是语法级硬约束,无例外。
_helper)模拟私有:它仍是导出的(因为 _ 不是字母,但 Go 视首字符为 _ 时仍判定为未导出;不过这是特例,不推荐依赖)// ValidateEmail checks format and returns error if invalid)go vet 或 staticcheck 可检测未使用的导出标识符,帮助精简 API