Go通过接口+结构体+函数值实现工厂与适配器模式:定义Logger接口统一契约,ZapLoggerAdapter和FileWriterAdapter等适配异构日志组件,NewLogger工厂按配置创建对应实例,解耦创建逻辑与具体类型。
用 Go 实现工厂 + 适配器模式,核心是解耦对象创建逻辑与具体类型,同时让不兼容的接口“变相”符合统一契约。Go 没有类和继承,但靠接口(interface)+ 结构体(struct)+ 函数值,完全可以达成相同设计意图。
先确定对外暴露的抽象能力——比如一个通用的 Logger 接口:
type Logger interface {
Log(msg string)
}
所有日志实现都必须满足这个契约,后续才能被统一使用。
现实里你可能已有第三方日志库(如 zap.Logger)或旧系统模块(如自定义 FileWriter),它们类型不同、方法名不同、参数也不同。这时就需要适配器:
zap.Logger,把 Info() 映射为 Log()
WriteLine(string) 封装成 Log(string)
type ZapLoggerAdapter struct {
logger *zap.Logger
}
func (a *ZapLoggerAdapter) Log(msg string) {
a.logger.Info(msg)
}
type FileWriterAdapter struct {
file *os.File
}
func (a *FileWriterAdapter) Log(msg string) {
a.file.WriteString(msg + "\n")
}
避免调用方感知底层类型差异,提供一个配置驱动的工厂函数:
type LoggerType string
const (
ConsoleLogger LoggerType = "console"
FileLogger LoggerType = "file"
ZapLogger LoggerType = "zap"
)
func NewLogger(t LoggerType, cfg interface{}) (Logger, error) {
switch t {
case ConsoleLogger:
return &ConsoleLoggerImpl{}, nil
case FileLogger:
if path, ok := cfg.(string); ok {
f, err := os.OpenFile(path, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)
if err != nil {
return nil, err
}
return &FileWriterAdapter{file: f}, nil
}
case ZapLogger:
z, err := zap.NewDevelopment()
if err != nil {
return nil, err
}
return &ZapLoggerAdapter{logger: z}, nil
}
return nil, fmt.Errorf("unknown logger type: %s", t)
}
调用时只需传入类型和配置,完全不关心内部结构是否带 zap 或 os.File:
logger, _ := NewLogger(FileLogger, "/tmp/app.log")
logger.Log("service started") // 统一调用
如果某些老模块只提供函数而非结构体(比如一个全局 func PrintToDB(string)),也可适配:
type FuncLoggerAdapter struct {
fn func(string)
}
func (a *FuncLoggerAdapter) Log(msg string) {
a.fn(msg)
}
// 工厂中支持:
case "db":
return &FuncLoggerAdapter{fn: PrintToDB}, nil
Go 的函数是一等公民,这种轻量适配非常自然,无需额外 wrapper 类。
不复杂但容易忽略:适配器本身不新增功能,只做“翻译”;工厂不持有实例状态,保持无副作用。两者配合,就能在不改旧代码的前提下,让新系统平滑接入各种异构组件。