0

0

Golang Decorator装饰器模式功能扩展示例

P粉602998670

P粉602998670

发布时间:2025-09-29 08:07:01

|

758人浏览过

|

来源于php中文网

原创

Go语言中通过接口组合和函数包装实现装饰器模式,用于日志、监控等功能扩展。首先定义UserService接口及其实现,接着创建loggingDecorator和metricsDecorator结构体,分别实现调用前后的日志记录与耗时统计。通过NewLoggingDecorator和NewMetricsDecorator逐层包装服务实例,形成责任链。装饰顺序影响执行流程,如先日志后指标,则日志不包含耗时本身时间。此外,可采用函数式装饰器,利用UserFunc类型和WithLogging、WithMetrics高阶函数实现更灵活的轻量级装饰,适用于中间件场景。最终通过组合方式实现关注点分离,提升代码可维护性。

golang decorator装饰器模式功能扩展示例

在 Go 语言中,装饰器模式通常通过函数包装或接口组合的方式来实现功能扩展,而不改变原有结构的代码。它广泛应用于日志记录、权限校验、缓存、监控等场景。下面通过一个清晰的示例展示如何使用装饰器模式为服务添加日志和耗时统计功能。

定义基础服务接口

假设我们有一个用户服务,提供获取用户信息的方法:

type UserService interface {
    GetUser(id int) string
}

type userService struct{}

func (s *userService) GetUser(id int) string {
    fmt.Printf("Fetching user with ID: %d\n", id)
    return fmt.Sprintf("User-%d", id)
}

创建装饰器结构体

我们可以定义一个装饰器结构体,它内部持有 UserService 接口实例,从而可以在调用前后插入额外逻辑:

type loggingDecorator struct {
    service UserService
}

func NewLoggingDecorator(service UserService) UserService {
    return &loggingDecorator{service: service}
}

func (d *loggingDecorator) GetUser(id int) string {
    fmt.Printf("[LOG] Getting user with ID: %d\n", id)
    result := d.service.GetUser(id)
    fmt.Printf("[LOG] Got result: %s\n", result)
    return result
}

添加性能监控装饰器

再封装一个统计执行时间的装饰器:

立即学习go语言免费学习笔记(深入)”;

type metricsDecorator struct {
    service UserService
}

func NewMetricsDecorator(service UserService) UserService {
    return &metricsDecorator{service: service}
}

func (d *metricsDecorator) GetUser(id int) string {
    start := time.Now()
    result := d.service.GetUser(id)
    elapsed := time.Since(start)
    fmt.Printf("[METRICS] GetUser(%d) took %v\n", id, elapsed)
    return result
}

组合多个装饰器

Go 支持将多个装饰器逐层包装,形成责任链式的处理流程:

func main() {
    var service UserService = &userService{}

    // 装饰:先加日志,再加指标
    service = NewLoggingDecorator(service)
    service = NewMetricsDecorator(service)

    // 调用方法
    service.GetUser(42)
}

输出结果类似:

环保节能智能空气净化器类网站模板(带手机版)1.4.2
环保节能智能空气净化器类网站模板(带手机版)1.4.2

环保节能智能空气净化器类网站模板(带手机版)安装即用,自带人人站CMS内核,支持移动端,前端banner轮播图文本均已进行可视化配置,伪静态页面生成,支持内容模型,支持多种URL模式及模型。模板特点:1、安装即用,自带人人站CMS内核及企业站展示功能(产品,新闻,案例展示等),并可根据需要增加表单 搜索等功能(自带模板) 2、支持响应式 3、前端banner轮播图文本均已进行可视化配置 4、伪静态

下载
[LOG] Getting user with ID: 42
Fetching user with ID: 42
[LOG] Got result: User-42
[METRICS] GetUser(42) took 12.5µs

注意装饰顺序会影响执行流程。如果交换装饰器顺序,日志将不包含耗时打印本身的时间。

函数式装饰器(可选高级写法)

对于更轻量的场景,可以使用函数式方式实现装饰器:

type UserFunc func(int) string

func (f UserFunc) GetUser(id int) string {
    return f(id)
}

func WithLogging(fn UserFunc) UserFunc {
    return func(id int) string {
        fmt.Printf("[LOG] Call GetUser(%d)\n", id)
        result := fn(id)
        fmt.Printf("[LOG] Result: %s\n", result)
        return result
    }
}

func WithMetrics(fn UserFunc) UserFunc {
    return func(id int) string {
        start := time.Now()
        result := fn(id)
        fmt.Printf("[METRICS] Took %v\n", time.Since(start))
        return result
    }
}

使用方式:

var getUser UserFunc = func(id int) string {
    return fmt.Sprintf("User-%d", id)
}

getUser = WithLogging(WithMetrics(getUser))
getUser(42)

这种方式更灵活,适合中间件类逻辑,如 HTTP 处理器链。

基本上就这些。Go 没有泛型装饰器语法糖,但通过接口和组合能自然实现装饰器模式,干净地分离关注点,提升代码可维护性。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

180

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

228

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

340

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

209

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

393

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

197

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

191

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

253

2025.06.17

C++ 高级模板编程与元编程
C++ 高级模板编程与元编程

本专题深入讲解 C++ 中的高级模板编程与元编程技术,涵盖模板特化、SFINAE、模板递归、类型萃取、编译时常量与计算、C++17 的折叠表达式与变长模板参数等。通过多个实际示例,帮助开发者掌握 如何利用 C++ 模板机制编写高效、可扩展的通用代码,并提升代码的灵活性与性能。

4

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 4.1万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号