基于Gin封装Web框架 - 10. 使用 context 上下文完成依赖注入

简介: 基于Gin封装Web框架 - 10. 使用 context 上下文完成依赖注入

基于Gin封装Web框架 - 10. 使用 context 上下文完成依赖注入

源码参考: https://github.com/go-jarvis/gin-rum

在开发过程中, 不可避免的会用到诸如 数据库、redis 等其他组件。 使用 依赖注入 的方式可以很好的对程序进行解耦。

选择 context 作为容器

之所以选择 Context 作为容器,

其一 , context 具有很强的裂变性,不同 RumGroup 的 context 可以添加属于自己的内容;

// github.com/go-jarvis/gin-rum/rum/rumgroup.go

// baseRumGroup 通过 Rum 返回一个根 RumGroup
func baseRumGroup(ctx context.Context, r *Rum, group string) *RumGroup {
    return &RumGroup{
        RouterGroup: r.RouterGroup.Group(group),
        ctx:         ctx,
    }
}

// newRumGroup 通过 RumGroup 扩展新的 RumGroup
func newRumGroup(base *RumGroup, group string) *RumGroup {
    return &RumGroup{
        RouterGroup: base.RouterGroup.Group(group),
        ctx:         base.ctx,
    }
}

// WithContext 向 RumGoft 中注入任何内容
// 以向 class 控制器传递
func (grp *RumGroup) WithContext(fns ...ContextFunc) {
    for _, fn := range fns {
        grp.ctx = fn(grp.ctx)
    }
}
AI 代码解读

其二 , context 的包容性, key , value 都是 interface{} 可以包容一切。

// context/context.go
func WithValue(parent Context, key, val interface{}) Context {
    if parent == nil {
        panic("cannot create context from nil parent")
    }
    if key == nil {
        panic("nil key")
    }
    if !reflectlite.TypeOf(key).Comparable() {
        panic("key is not comparable")
    }
    return &valueCtx{parent, key, val}
}
AI 代码解读

rum 增加上下文支持

随后, 在 RumGroup 中增加 context.Context 字段,并添加相关方法。

type RumGroup struct {
    *gin.RouterGroup
    ctx context.Context
}

type ContextFunc = func(context.Context) context.Context

// WithContext 向 RumGoft 中注入任何内容
// 以向 class 控制器传递
func (grp *RumGroup) WithContext(fns ...ContextFunc) {
    for _, fn := range fns {
        grp.ctx = fn(grp.ctx)
    }
}
AI 代码解读

WithContext 方法要求传入的是一个 操作 Context 的函数, 这个函数由用户自己实现。WithContext 方法对于 Context 的修改仅限于 RumGroup 自身与及其子 RumGroup。

另外, 在 ClassController 中, 也需要做响应的变更, 需要 Handler(ctx context.Context) 方法支持 context 作为参数传递。

type ClassController interface {
    Method() string
    Path() string

    // Handler() (interface{}, error)  // 老方法
    Handler(context.Context) (interface{}, error)
}
AI 代码解读

为什么不用 gin.Context

虽然 gin.Context 也实现了 context.Context 的接口, 在在和我们常用的 标准库 还是还是有很多差别

首先, gin.Context 在 gin 初始化的时候会生成一个 公共的祖先 gin.Context, 随着程序的启动, 用户每次请求都将创建一个独立的 gin.Context 副本。

由于 gin 并没有提供一个可以用户初始化的 gin.Context 的 API。

func New() *Engine {
    debugPrintWARNINGNew()
    engine := &Engine{
        // ...省略
    }
    engine.RouterGroup.engine = engine
    // pool 是 engine 的私有字段
    engine.pool.New = func() interface{} {
        return engine.allocateContext()
    }
    return engine
}

// allocateContext 也是 engine 的私有方法
func (engine *Engine) allocateContext() *Context {
    v := make(Params, 0, engine.maxParams)
    skippedNodes := make([]skippedNode, 0, engine.maxSections)
    return &Context{engine: engine, params: &v, skippedNodes: &skippedNodes}
}
AI 代码解读

因此用户只能写入到 每次请求 的 gin.Context 中。
而类似 数据库连接池 这样的句柄, 在程序启动的时候就初始化了,不在会改变。 如果写入到 gin.Context 中就造成了计算资源的浪费。

其次, 在 gin.Context.Value() 方法首先与标准库的实现不同, 有一个 比较的致命问题

// github.com/gin-gonic/gin@v1.7.4/context.go
func (c *Context) Value(key interface{}) interface{} {
    if key == 0 {
        return c.Request
    }
    if keyAsString, ok := key.(string); ok {
        val, _ := c.Get(keyAsString)
        return val
    }
    return nil
}
AI 代码解读

可以看到, gin 中的 Value() 方法将 key 转为了字符串。 因为此失去了 数据类型 的支持, key 的唯一性概率就大大降低了, 很容易发生覆盖冲突。

这一点在标准库中的 valueCtx 就不存在这种情况, 因为 key 不会被断言, 类型也是一部分

func (c *valueCtx) Value(key interface{}) interface{} {
    if c.key == key {
        return c.val
    }
    return c.Context.Value(key)
}
AI 代码解读

测试代码参考 context-Context-and-gin-Context

遗留问题

由于使用了自建的 GroupGroup Context, 并且 gin.Context 没有交叉点。 因此 gin.Context 中与 Cancel 相关的方法也无法传递到 ClassController 中的 Handler 方法中。

demo: 使用 context 传递

gorm 数据库句柄 通过 Context 注入到 Rum 中。

封装 db adaptor

首先, 创建新的数据类型, 并用该类型创建 唯一 key

type contextGormDBType int

var contextGormDBKey contextGormDBType = 0
AI 代码解读

在实践中, 不同的适配器可以创建自己的数据类型。 如此一来, 即使 字面值 相同也不会冲突、覆盖。

其次, 实现 注入提取 函数。

// WithGormDB 注入 *gorm.DB 到 context 中
func WithGormDB(vaule *gorm.DB) func(context.Context) context.Context {
    return func(ctx context.Context) context.Context {
        return context.WithValue(ctx, contextGormDBKey, value)
    }
}

// FromContextGormDB 从 context 中提取 *gorm.DB
func FromContextGormDB(ctx context.Context) *gorm.DB {
    return ctx.Value(contextGormDBKey).(*gorm.DB)
}
AI 代码解读

在 rum 中注入 adaptor

func main() {

    // 1. 使用 rum 代替 gin
    g := rum.Default()
    g.WithContext(
        db.WithGormDB(db.NewGormDB()),
    )

// 省略
AI 代码解读

在 ClassController 中使用 adaptor

最后, 在 ClassController 实例中直接使用。

注意, 在控制器定义的时候依旧保持 干净, 无任何依赖适配器的字段。

// GetUserByID class 控制器
type GetUserByID struct {
    httpx.MethodPost

    UserID int `uri:"id"`
}

func (user *GetUserByID) Handler(ctx context.Context) (interface{}, error) {
    // 获取 ctx 中注入的 *gorm.DB 对象
    gorm := db.FromContextGormDB(ctx)

    userModel := &User{}
    tx := gorm.Where("user_id=?", user.UserID).First(&userModel)

    return userModel, tx.Error
}
AI 代码解读
目录
打赏
0
0
0
0
7
分享
相关文章
Go语言Web开发框架实践:路由、中间件、参数校验
Gin框架以其极简风格、强大路由管理、灵活中间件机制及参数绑定校验系统著称。本文详解其核心功能:1) 路由管理,支持分组与路径参数;2) 中间件机制,实现全局与局部控制;3) 参数绑定,涵盖多种来源;4) 结构体绑定与字段校验,确保数据合法性;5) 自定义校验器扩展功能;6) 统一错误处理提升用户体验。Gin以清晰模块化、流程可控及自动化校验等优势,成为开发者的优选工具。
Go语言Web开发框架实践:使用 Gin 快速构建 Web 服务
Gin 是一个高效、轻量级的 Go 语言 Web 框架,支持中间件机制,非常适合开发 RESTful API。本文从安装到进阶技巧全面解析 Gin 的使用:快速入门示例(Hello Gin)、定义 RESTful 用户服务(增删改查接口实现),以及推荐实践如参数校验、中间件和路由分组等。通过对比标准库 `net/http`,Gin 提供更简洁灵活的开发体验。此外,还推荐了 GORM、Viper、Zap 等配合使用的工具库,助力高效开发。
|
3月前
|
Golang | Gin:net/http与Gin启动web服务的简单比较
总的来说,`net/http`和 `Gin`都是优秀的库,它们各有优缺点。你应该根据你的需求和经验来选择最适合你的工具。希望这个比较可以帮助你做出决策。
105 35
测试工程师要失业?Magnitude:开源AI Agent驱动的端到端测试框架,让Web测试更智能,自动完善测试用例!
Magnitude是一个基于视觉AI代理的开源端到端测试框架,通过自然语言构建测试用例,结合推理代理和视觉代理实现智能化的Web应用测试,支持本地运行和CI/CD集成。
402 15
测试工程师要失业?Magnitude:开源AI Agent驱动的端到端测试框架,让Web测试更智能,自动完善测试用例!
WEB CAD插件通过上下文对象MxPluginContext修改UI界面的方法
本文介绍了如何使用MxPluginContext动态控制MxCAD项目的UI界面。通过该上下文对象,开发者可以灵活设置UI配置,如控制操作栏显隐、编辑按钮、添加侧边栏等。具体方法包括调用`getUiConfig()`获取并修改`mxUiConfig.json`中的属性,实现界面的定制化。此外,还提供了控制命令行聚焦的功能,解决输入框焦点锁定问题。详细代码示例和效果对比图展示了具体实现步骤,帮助开发者更好地适配项目需求。
TypeScript 是一种静态类型的编程语言,它扩展了 JavaScript,为 Web 开发带来了强大的类型系统、组件化开发支持、与主流框架的无缝集成、大型项目管理能力和提升开发体验等多方面优势
TypeScript 是一种静态类型的编程语言,它扩展了 JavaScript,为 Web 开发带来了强大的类型系统、组件化开发支持、与主流框架的无缝集成、大型项目管理能力和提升开发体验等多方面优势。通过明确的类型定义,TypeScript 能够在编码阶段发现潜在错误,提高代码质量;支持组件的清晰定义与复用,增强代码的可维护性;与 React、Vue 等框架结合,提供更佳的开发体验;适用于大型项目,优化代码结构和性能。随着 Web 技术的发展,TypeScript 的应用前景广阔,将继续引领 Web 开发的新趋势。
151 2
Django框架适合开发哪种类型的Web应用程序?
Django 框架凭借其强大的功能、稳定性和可扩展性,几乎可以适应各种类型的 Web 应用程序开发需求。无论是简单的网站还是复杂的企业级系统,Django 都能提供可靠的支持,帮助开发者快速构建高质量的应用。同时,其活跃的社区和丰富的资源也为开发者在项目实施过程中提供了有力的保障。
293 62
Go语言中几种流行的Web框架,如Beego、Gin和Echo,分析了它们的特点、性能及适用场景,并讨论了如何根据项目需求、性能要求、团队经验和社区支持等因素选择最合适的框架
本文概述了Go语言中几种流行的Web框架,如Beego、Gin和Echo,分析了它们的特点、性能及适用场景,并讨论了如何根据项目需求、性能要求、团队经验和社区支持等因素选择最合适的框架。
731 1
从框架到现代Web开发实践
从框架到现代Web开发实践
115 1
|
8月前
|
探索PHP的现代演进:从Web开发到框架创新
PHP是一种流行的服务器端脚本语言,自诞生以来在Web开发领域占据重要地位。从简单的网页脚本到支持面向对象编程的现代语言,PHP经历了多次重大更新。本文探讨PHP的现代演进历程,重点介绍其在Web开发中的应用及框架创新,如Laravel、Symfony等。这些框架不仅简化了开发流程,还提高了开发效率和安全性。
72 3

热门文章

最新文章

AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等