代理模式在Go中通过组合与函数封装实现,核心是代理对象持有真实对象引用并在调用前后插入控制逻辑;支持结构体嵌入、函数装饰器及接口委托三种方式,适用于鉴权、懒加载、熔断等场景。

代理模式在 Go 中不依赖继承或接口强制约束,而是通过组合与函数封装自然实现——核心是让代理对象持有真实对象引用,并在调用前后插入控制逻辑。
用结构体封装真实对象,实现访问拦截
Go 没有传统面向对象的“方法重写”,但可通过嵌入(embedding)+ 显式方法定义轻松模拟代理行为。关键在于:代理结构体包含真实对象字段,且对外暴露相同方法签名,在方法体内决定是否调用、何时调用、如何包装结果。
- 定义一个 RealSubject 类型,包含业务逻辑(如
DoWork()) - 定义 Proxy 结构体,内嵌或字段持有
*RealSubject - 在
Proxy.DoWork()中添加权限检查、日志、缓存判断等前置/后置逻辑,再有条件地调用r.DoWork()
用函数类型实现轻量级代理(无结构体开销)
当只需拦截单一方法或临时增强行为时,可直接用函数值作为代理。Go 的函数是一等公民,天然支持装饰器风格。
- 声明类型
type Operation func() error - 编写装饰函数:
func WithLogging(op Operation) Operation,返回新函数,在调用op()前后打印日志 - 也可叠加多个装饰器:
WithAuth(WithLogging(realOp))
接口 + 代理组合,支持运行时替换
若需灵活切换真实对象或统一抽象,可先定义接口(如 Subject),让 RealSubject 和 Proxy 都实现它。这样上层代码只依赖接口,代理可透明介入。
立即学习“go语言免费学习笔记(深入)”;
- 接口定义行为契约,不暴露实现细节
-
Proxy实现同一接口,内部委托给具体实例 - 初始化时传入真实对象(或工厂函数),便于单元测试中注入 mock
常见控制场景示例:鉴权、延迟加载、日志与熔断
代理的价值在于解耦横切关注点。例如:
-
鉴权代理:调用前检查
ctx.Value("user").(*User).HasRole("admin"),不满足则直接返回错误 -
懒加载代理:
Proxy字段初始为nil,首次调用时才初始化真实对象并缓存 - 熔断代理:维护失败计数和时间窗口,连续失败超阈值时跳过真实调用,快速返回 fallback










