0

0

如何使用Golang实现装饰器+代理模式_增强功能和访问控制

P粉602998670

P粉602998670

发布时间:2026-01-11 16:47:02

|

993人浏览过

|

来源于php中文网

原创

Go 语言虽无语法级装饰器,但可通过高阶函数模拟中间件(如日志、鉴权)、结构体+接口实现代理模式,并结合 context 与泛型提升复用性与类型安全。

如何使用golang实现装饰器+代理模式_增强功能和访问控制

Go 语言本身不支持 Python 那样的语法级装饰器(如 @decorator),但可以通过高阶函数、接口抽象和组合方式,自然地模拟“装饰器 + 代理模式”的行为——既能增强功能(如日志、重试、熔断),又能控制访问(如权限校验、限流、鉴权)。关键在于:用函数包装函数,用结构体包装接口,让增强逻辑与业务逻辑解耦。

用高阶函数实现装饰器风格的中间件

将核心逻辑抽象为函数类型(如 func(context.Context, Request) Response),再编写接受该函数并返回新函数的“装饰器”:

  • 定义统一处理签名:type HandlerFunc func(ctx context.Context, req interface{}) (interface{}, error)
  • 写日志装饰器:
    func WithLogging(next HandlerFunc) HandlerFunc {
    return func(ctx context.Context, req interface{}) (interface{}, error) {
    log.Printf("→ %T received", req)
    res, err := next(ctx, req)
    log.Printf("← completed with error: %v", err)
    return res, err
    }
    }
  • 链式组合:
    handler := WithLogging(WithAuth(WithRateLimit(myBusinessHandler)))
    执行时从外到内(装饰器)→ 内到外(返回)

用结构体+接口实现代理模式(静态代理)

当需要更精细控制(如字段级拦截、动态行为切换)时,定义接口 + 真实实现 + 代理结构体:

  • 定义服务接口:type UserService interface { GetUser(id int) (*User, error); CreateUser(u *User) error }
  • 真实实现:type RealUserService struct{} 实现全部方法
  • 代理结构体持有一个 UserService 字段:
    type UserProxy struct { svc UserService; authChecker AuthChecker }
    GetUser 中先调 authChecker.Check("read:user"),通过才委托给 svc.GetUser
  • 优势:可嵌套代理(如 LoggingProxy{UserProxy{RealUserService{}}}),也便于单元测试(注入 mock svc)

结合 context 实现运行时访问控制

装饰器或代理中常需获取请求上下文信息(如用户身份、租户 ID、API Key)。推荐将认证信息注入 context.Context,而非靠参数传递:

GitHub Copilot
GitHub Copilot

GitHub AI编程工具,实时编程建议

下载

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

  • 中间层解析 token 后:ctx = context.WithValue(ctx, userCtxKey, &User{ID: 123, Role: "admin"})
  • 后续装饰器或代理直接从 ctx 取:user, ok := ctx.Value(userCtxKey).(*User)
  • 权限检查示例:
    if !user.Can("delete:user") { return nil, errors.New("forbidden") }
  • 注意:避免滥用 context.WithValue;建议定义强类型 key(type userCtxKey struct{})防止 key 冲突

进阶:用泛型简化通用装饰器(Go 1.18+)

避免为每种 handler 类型重复写装饰器。利用泛型统一抽象:

  • 定义泛型处理器type Handler[T any, R any] func(context.Context, T) (R, error)
  • 泛型日志装饰器:
    func LogHandler[T, R any](next Handler[T, R]) Handler[T, R] {
    return func(ctx context.Context, req T) (R, error) {
    log.Printf("handling %T", req)
    return next(ctx, req)
    }
    }
  • 调用时自动推导:h := LogHandler(processPayment)(假设 processPaymentHandler[PayReq, PayResp]

相关专题

更多
python开发工具
python开发工具

php中文网为大家提供各种python开发工具,好的开发工具,可帮助开发者攻克编程学习中的基础障碍,理解每一行源代码在程序执行时在计算机中的过程。php中文网还为大家带来python相关课程以及相关文章等内容,供大家免费下载使用。

745

2023.06.15

python打包成可执行文件
python打包成可执行文件

本专题为大家带来python打包成可执行文件相关的文章,大家可以免费的下载体验。

634

2023.07.20

python能做什么
python能做什么

python能做的有:可用于开发基于控制台的应用程序、多媒体部分开发、用于开发基于Web的应用程序、使用python处理数据、系统编程等等。本专题为大家提供python相关的各种文章、以及下载和课程。

757

2023.07.25

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

617

2023.07.31

python教程
python教程

Python已成为一门网红语言,即使是在非编程开发者当中,也掀起了一股学习的热潮。本专题为大家带来python教程的相关文章,大家可以免费体验学习。

1260

2023.08.03

python环境变量的配置
python环境变量的配置

Python是一种流行的编程语言,被广泛用于软件开发、数据分析和科学计算等领域。在安装Python之后,我们需要配置环境变量,以便在任何位置都能够访问Python的可执行文件。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

547

2023.08.04

python eval
python eval

eval函数是Python中一个非常强大的函数,它可以将字符串作为Python代码进行执行,实现动态编程的效果。然而,由于其潜在的安全风险和性能问题,需要谨慎使用。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

577

2023.08.04

scratch和python区别
scratch和python区别

scratch和python的区别:1、scratch是一种专为初学者设计的图形化编程语言,python是一种文本编程语言;2、scratch使用的是基于积木的编程语法,python采用更加传统的文本编程语法等等。本专题为大家提供scratch和python相关的文章、下载、课程内容,供大家免费下载体验。

705

2023.08.11

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

80

2026.01.09

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 0.6万人学习

Django 教程
Django 教程

共28课时 | 3万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.1万人学习

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

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