基于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)
    }
}

其二 , 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}
}

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)
    }
}

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)
}

为什么不用 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}
}

因此用户只能写入到 每次请求 的 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
}

可以看到, 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)
}

测试代码参考 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

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

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

// 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)
}

在 rum 中注入 adaptor

func main() {

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

// 省略

在 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
}
相关文章
|
1月前
|
中间件 API Go
使用Echo和Gin构建高性能Web服务的技术文档
本文档对比了Go语言中的两个流行Web框架——Echo和Gin。Echo是一个高性能、可扩展的框架,适合构建微服务和API,强调简洁API和并发性能。Gin基于net/http包,具有Martini风格API,以其快速路由和丰富社区支持闻名。在性能方面,Gin的路由性能出色,两者并发性能均强,内存占用低。文中还提供了使用Echo和Gin构建Web服务的代码示例,帮助开发者了解如何运用这两个框架。选择框架应考虑项目需求和个人喜好。
48 2
|
27天前
|
关系型数据库 MySQL 数据库
如何使用Python的Flask框架来构建一个简单的Web应用
如何使用Python的Flask框架来构建一个简单的Web应用
39 0
|
17天前
|
开发框架 中间件 PHP
Laravel框架:优雅构建PHP Web应用的秘诀
**Laravel 框架简介:** Laravel是PHP的优雅Web开发框架,以其简洁语法、强大功能和良好开发者体验闻名。它强调代码的可读性和可维护性,加速复杂应用的构建。基础步骤包括安装PHP和Composer,然后运行`composer create-project`创建新项目。Laravel的路由、控制器和Blade模板引擎简化了HTTP请求处理和视图创建。模型和数据库迁移通过Eloquent ORM使数据库操作直观。Artisan命令行工具、队列、事件和认证系统进一步增强了其功能。【6月更文挑战第26天】
19 1
|
18天前
|
JavaScript 前端开发 开发者
Angular框架:企业级Web应用的强大后盾
Angular,谷歌支持的JavaScript框架,因其组件化架构、双向数据绑定、依赖注入和路由系统,成为企业级Web开发首选。组件化促进代码重用,如`AppComponent`示例。双向数据绑定简化DOM操作,减少手动工作。依赖注入通过示例展示易管理依赖,提升测试性。路由则支持SPA开发,平滑页面过渡。Angular的特性增强了开发效率和应用质量,使其在Web开发领域保持领先地位。【6月更文挑战第25天】
22 2
|
19天前
|
前端开发 数据库 开发者
构建高效后端:Django框架在Web开发中的深度解析
**Django框架深度解析摘要** Django,Python的高级Web框架,以其快速开发和简洁设计备受青睐。核心特性包括Model-Template-View架构、ORM、模板引擎和URL路由。通过创建博客应用示例,展示从初始化项目、定义模型、创建视图和URL配置到使用模板的流程,体现Django如何简化开发,提高效率。其强大功能如用户认证、表单处理等,使Django成为复杂Web应用开发的首选。学习Django,提升Web开发效率。【6月更文挑战第24天】
52 1
|
1天前
|
安全 API 数据库
Django/Flask不只是框架,它们是你Web开发路上的超级英雄!
【7月更文挑战第14天】Django与Flask,Python Web开发的双雄。Django提供全面功能,如ORM、模板引擎,适合大型项目;Flask轻量灵活,适用于快速迭代的定制化应用。Django示例展示ORM简化数据库操作,Flask示例演示构建RESTful API的便捷。两者各有所长,为开发者创造无限可能。**
|
5天前
|
存储 JavaScript 前端开发
WEB三大主流框架之Angular
WEB三大主流框架之Angular
|
5天前
|
存储 JavaScript 前端开发
WEB三大主流框架之Vue.js
WEB三大主流框架之Vue.js
|
5天前
|
XML 前端开发 JavaScript
WEB三大主流框架之React
WEB三大主流框架之React
|
1月前
|
前端开发 JavaScript 测试技术
web前端语言框架:探索现代前端开发的核心架构
web前端语言框架:探索现代前端开发的核心架构
23 4