Go中无私有包概念,但可通过小写标识符控制导出、internal目录限制导入、接口抽象隐藏实现、避免internal单独发布四种方式实现封装。
在 Go 中,私有包本身并不存在——Go 没有像 Java 或 C# 那样的访问修饰符(如 private、protected)。但你可以通过包级作用域可见性规则和目录结构设计来有效封装内部实现,阻止外部直接访问非公开逻辑。
Go 的可见性由标识符首字母决定:大写(如 MyFunc)表示导出(public),小写(如 myFunc)表示未导出(private to the package)。这是最基础也是最关键的封装机制。
myHelper() 或访问 defaultConfig
Go 官方推荐使用 internal/ 目录来存放仅限特定父级路径使用的代码。构建工具(如 go build)会强制检查:只有 internal 的直接父目录及其子目录下的包才能导入它。
myproject/internal/dbutil/ 和 myproject/internal/authz/
myproject/cmd/api/ 可以导入 myproject/internal/dbutil
github.com/others/project 尝试导入会报错:use of internal package not allowed
internal 是约定而非语法,依赖 Go 工具链的路径检查对外只导出接口类
型,把具体结构体和构造函数设为小写,迫使调用方只能通过工厂函数或配置函数获取实例,且无法直接修改内部状态。
type Cache interface { Get(key string) (any, bool) }
type redisCache struct { client *redis.Client }
func NewCache(opts ...Option) Cache
redisCache.client 等细节如果你用 go mod 管理依赖,确保 internal 子目录不被单独发布为可 go-get 的模块。模块路径应指向顶层(如 example.com/myapp),而不是 example.com/myapp/internal。
go.mod 中声明主模块路径为根目录,不要为 internal/ 下的子目录创建额外 module 声明go get example.com/myapp/internal/xxx 这类错误示例internal 包的 doc.go 中加注释说明:“仅供本项目内部使用,禁止外部导入”