Go中代理模式通过组合接口和结构体字段实现,代理类型需实现相同接口并在方法中控制调用逻辑;需注意避免nil panic、显式转发、用recover()捕获panic、用context.WithTimeout()控制超时。
Go 语言没有类继承和接口强制实现机制,所以代理模式不靠“继承父类 + 重写方法”来实现,而是通过组合 interface{} 和结构体字段封装被代理对象。关键在于:代理类型必须实现与被代理对象相同的接口,且在方法中控制调用时机、参数、返回值或是否转发。
典型结构是:
Service),包含业务方法(如 Do())RealService)实现该接口AuthProxy)持有 Service 接口字段,并实现相同接口,在方法内插入访问控制逻辑这是最常用、最符合 Go 风格的做法。代理不隐藏原始对象,而是显式组合并控制其行为。常见错误是直接代理指针导致 nil panic,或忘记在代理方法中调用底层 Do()。
示例场景:只允许 admin 用户调用 Do() 方法:
立即学习“go语言免费学习笔记(深入)”;
type Service interface {
Do() string
}
type RealService
struct{}
func (r *RealService) Do() string {
return "real work done"
}
type AuthProxy struct {
service Service
user string
}
func (a *AuthProxy) Do() string {
if a.user != "admin" {
return "access denied"
}
return a.service.Do() // 必须显式调用,否则不转发
}
// 使用:
svc := &RealService{}
proxy := &AuthProxy{service: svc, user: "guest"}
fmt.Println(proxy.Do()) // "access denied"
真实服务可能阻塞或 panic,代理需兜底。Go 中不能像 Java 那样用 try-catch,但可用 recover() 捕获 panic,用 context.WithTimeout() 控制执行时间。忽略这两点会导致代理失效,变成单点故障。
recover() 必须在 defer 中调用,且仅对当前 goroutine 有效context.Context 注入被代理方法(即修改接口签名),或用 time.AfterFunc 异步中断(不推荐,难清理)改进接口后示例:
type ServiceWithContext interface {
Do(ctx context.Context) (string, error)
}
func (a *AuthProxy) Do(ctx context.Context) (string, error) {
select {
case <-ctx.Done():
return "", ctx.Err()
default:
}
if a.user != "admin" {
return "", errors.New("access denied")
}
// 调用前设置子 context 防止泄漏
subCtx, cancel := context.WithTimeout(ctx, 2*time.Second)
defer cancel()
// 假设 realService 支持 context
return a.service.(ServiceWithContext).Do(subCtx)
}
有人会写 func(Service) Service 类型的包装器,例如:
func WithAuth(s Service, user string) Service {
return &authWrapper{service: s, user: user}
}
这看似简洁,但问题明显:
proxy.MaxRetries),失去可维护性fmt.Printf("%T", s) 显示匿名类型)真正需要灵活链式代理时,应使用结构体字段组合多个策略(如 AuthProxy 内嵌 RateLimitProxy),而不是靠闭包层层套娃。
代理模式的复杂点不在语法,而在责任边界——谁校验权限、谁处理超时、谁记录日志、谁决定重试。这些必须在代理结构体内部明确划分,而不是堆砌 if 判断。