golang beego结合wire依赖注入及自动路由

2024-04-30 10:20

本文主要是介绍golang beego结合wire依赖注入及自动路由,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1 安装wire

1.1 通过命令直接安装

go install github.com/google/wire/cmd/wire@latest

1.2 通过go get方式安装

go get github.com/google/wire/cmd/wire

进入目录编译

cd C:\Users\leell\go\pkg\mod\github.com\google\wire@v0.6.0\cmd\wire
go build

然后将wire.exe移动到 C:\Users\leell\go\bin目录下

具体目录通过go env进行查看

C:\Users\leell\go>go env
set GO111MODULE=on
set GOARCH=amd64
set GOBIN=C:\Users\leell\go\bin

2 配置业务代码

我没有按照beego的默认目录,而是通过自定义app包名进行业务模块化

2.1 分类模块

app/category/category_controller.go

type Controller struct {controller.BaseControllerservice *Service
}func NewController(service *Service) *Controller {return &Controller{service: service}
}

app/category/category_service.go

type Service struct {dao *Dao
}func NewService(dao *Dao) *Service {return &Service{dao}
}

app/category/category_dao.go 

type Dao struct {ormer orm.Ormer
}func NewDao(ormer orm.Ormer) *Dao {return &Dao{ormer}
}

app/category/fx.go

package categoryimport ("github.com/beego/beego/v2/server/web""github.com/google/wire"
)type FxRouter struct {controller *Controller
}func NewAppRouter(controller *Controller) *FxRouter {return &FxRouter{controller}
}
func (r *FxRouter) Router(namespace *web.Namespace) {namespace.Get("/category", r.controller.GetAll)namespace.Get("/category/:id", r.controller.GetOne)namespace.Post("/category", r.controller.Post)namespace.Put("/category/:id", r.controller.Put)namespace.Delete("/category/:id", r.controller.Delete)
}var Set = wire.NewSet(NewController, NewService, NewDao, NewAppRouter)

app/fx.go:将分类模块引入app/fx.go中并且等待注册路由

package appimport ("github.com/beego/beego/v2/server/web""github.com/google/wire""zhiqu/app/category""zhiqu/app/login""zhiqu/app/user"
)type FxRouter struct {categoryFxRouter *category.FxRouterloginFxRouter    *login.FxRouteruserFxRouter     *user.FxRouter
}func NewAppRouter(categoryFxRouter *category.FxRouter,loginFxRouter *login.FxRouter,userFxRouter *user.FxRouter) *FxRouter {return &FxRouter{categoryFxRouter, loginFxRouter, userFxRouter}
}
func (r *FxRouter) Router(namespace *web.Namespace) {r.categoryFxRouter.Router(namespace)r.loginFxRouter.Router(namespace)r.userFxRouter.Router(namespace)
}var Set = wire.NewSet(NewAppRouter, category.Set, login.Set, user.Set)

2.2 项目wire定义总提供者provider

infrastructure/database/pgsql/db.go

package pgsqlimport ("fmt""github.com/beego/beego/v2/client/orm"beego "github.com/beego/beego/v2/server/web"_ "github.com/lib/pq"
)func init() {pgsqlUser, _ := beego.AppConfig.String("pgsqlUser")pgsqlPass, _ := beego.AppConfig.String("pgsqlPass")pgsqlDbName, _ := beego.AppConfig.String("pgsqlDbName")pgsqlHost, _ := beego.AppConfig.String("pgsqlHost")pgsqlPort, _ := beego.AppConfig.String("pgsqlPort")pgsqlSslMode, _ := beego.AppConfig.String("pgsqlSslMode")// 注册驱动err := orm.RegisterDriver("postgres", orm.DRPostgres)if err != nil {panic(err)}// 设置默认数据库err = orm.RegisterDataBase("default", "postgres", fmt.Sprintf("user=%s password=%s dbname=%s host=%s port=%s sslmode=%s", pgsqlUser, pgsqlPass, pgsqlDbName, pgsqlHost, pgsqlPort, pgsqlSslMode))if err != nil {panic(err)}
}
func NewPgsqlOrm() orm.Ormer {ormer := orm.NewOrm()return ormer
}

provider/provider.go

var Set = wire.NewSet(NewBeanFactory, pgsql.NewPgsqlOrm, app.Set)

2.3 给项目一个bean工厂

这点有点像spring的beanfactory,这里主要是:

(1)解决beego路由一开始执行init能够得到所有controller的实例

(2)go项目中存在recycle的循环依赖,这里就可以用beanFactory做一个中间过度

package providerimport ("github.com/beego/beego/v2/core/logs""sync""zhiqu/app"
)var (AppBeanFactory *BeanFactoryonce           sync.OnceBeeLog         *logs.BeeLogger
)type BeanFactory struct {AppFxRouter *app.FxRouter
}func init() {BeeLog = logs.NewLogger()err := BeeLog.SetLogger(logs.AdapterConsole)if err != nil {return}
}
func NewBeanFactory(appFxRouter *app.FxRouter) *BeanFactory {once.Do(func() {AppBeanFactory = &BeanFactory{AppFxRouter: appFxRouter,}})return AppBeanFactory
}

3 wire配置代码及生成gen代码

 wire/wire.go

//go:build wireinject
// +build wireinjectpackage wireimport ("zhiqu/provider"
)
import "github.com/google/wire"// wire.go 初始化模块
func NewApp() (*provider.BeanFactory, error) {panic(wire.Build(provider.Set))
}

执行wire命令生成gen代码得到wire/wire_gen.go

wire gen wire/wire.go

生成最终代码文件内容

wire/wire_gen.go

// Code generated by Wire. DO NOT EDIT.//go:generate go run -mod=mod github.com/google/wire/cmd/wire
//go:build !wireinject
// +build !wireinjectpackage wireimport ("zhiqu/app/category""zhiqu/provider"
)// Injectors from wire.go:// wire.go 初始化模块
func NewApp() (*provider.BeanFactory, error) {categoryDao := category.NewDao()categoryService := category.NewService(categoryDao)categoryController := category.NewController(categoryService)beanFactory := provider.NewBeanFactory(categoryController)return beanFactory, nil
}

4 路由配置

routers/router.go

// Package routers 路由配置
package routersimport ("github.com/beego/beego/v2/server/web""zhiqu/wire"
)func init() {beanFactory, err := wire.NewApp()if err != nil {panic(err)}//这是一种可实现的依赖注入全自动方式ns := web.NewNamespace("/v1", func(namespace *web.Namespace) {beanFactory.AppFxRouter.Router(namespace)})web.SetStaticPath("/swagger", "swagger")web.AddNamespace(ns)
}

可以看出在路由文件中添加了wire生成的gen代码的获取的beanfactory工厂,然后将所有的单例go对象交由wire进行管理

注意:这里采用的是beego的web方式而不是beego方式,这里都是手动配置路由,不要去通过命令生成commentsRouter.go这样的路由文件

为啥不采用beego注册路由?

因为我们要想做到依赖注入,得确保controller、service、dao这些是单例,而beego注册的路由内部是现实每次收到请求都会通过反射技术重新构建一个controler进行处理。

正式因为最开始通过官方生成的项目配置路由注册时beego方式,而不能实现依赖注入,最终选择了gin+gorm+go.uber.org/fx的架构组合

5 查看最终结果

不过我不太推荐这种方式的依赖注入,虽然是在编译期实现了代码的生成,减少了启动通过反射来进行依赖注入的方式的时间,但是对开发人员来说多了一道工序。并且启动时间也要不了多久,而且编译器和运行期的依赖注入对接口运行是没有影响。

给大家推荐一块依赖注入框架:go.uber.org/fx

源代码:GitHub - leellun/beego-wire

这篇关于golang beego结合wire依赖注入及自动路由的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/948606

相关文章

每天认识几个maven依赖(ActiveMQ+activemq-jaxb+activesoap+activespace+adarwin)

八、ActiveMQ 1、是什么? ActiveMQ 是一个开源的消息中间件(Message Broker),由 Apache 软件基金会开发和维护。它实现了 Java 消息服务(Java Message Service, JMS)规范,并支持多种消息传递协议,包括 AMQP、MQTT 和 OpenWire 等。 2、有什么用? 可靠性:ActiveMQ 提供了消息持久性和事务支持,确保消

pip-tools:打造可重复、可控的 Python 开发环境,解决依赖关系,让代码更稳定

在 Python 开发中,管理依赖关系是一项繁琐且容易出错的任务。手动更新依赖版本、处理冲突、确保一致性等等,都可能让开发者感到头疼。而 pip-tools 为开发者提供了一套稳定可靠的解决方案。 什么是 pip-tools? pip-tools 是一组命令行工具,旨在简化 Python 依赖关系的管理,确保项目环境的稳定性和可重复性。它主要包含两个核心工具:pip-compile 和 pip

基于51单片机的自动转向修复系统的设计与实现

文章目录 前言资料获取设计介绍功能介绍设计清单具体实现截图参考文献设计获取 前言 💗博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师,一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP430/AVR等单片机设计 主要对象是咱们电子相关专业的大学生,希望您们都共创辉煌!✌💗 👇🏻 精彩专栏 推荐订阅👇🏻 单片机

Python3 BeautifulSoup爬虫 POJ自动提交

POJ 提交代码采用Base64加密方式 import http.cookiejarimport loggingimport urllib.parseimport urllib.requestimport base64from bs4 import BeautifulSoupfrom submitcode import SubmitCodeclass SubmitPoj():de

深入理解数据库的 4NF:多值依赖与消除数据异常

在数据库设计中, "范式" 是一个常常被提到的重要概念。许多初学者在学习数据库设计时,经常听到第一范式(1NF)、第二范式(2NF)、第三范式(3NF)以及 BCNF(Boyce-Codd范式)。这些范式都旨在通过消除数据冗余和异常来优化数据库结构。然而,当我们谈到 4NF(第四范式)时,事情变得更加复杂。本文将带你深入了解 多值依赖 和 4NF,帮助你在数据库设计中消除更高级别的异常。 什么是

PHP防止SQL注入详解及防范

SQL 注入是PHP应用中最常见的漏洞之一。事实上令人惊奇的是,开发者要同时犯两个错误才会引发一个SQL注入漏洞。 一个是没有对输入的数据进行过滤(过滤输入),还有一个是没有对发送到数据库的数据进行转义(转义输出)。这两个重要的步骤缺一不可,需要同时加以特别关注以减少程序错误。 对于攻击者来说,进行SQL注入攻击需要思考和试验,对数据库方案进行有根有据的推理非常有必要(当然假设攻击者看不到你的

PHP防止SQL注入的方法(2)

如果用户输入的是直接插入到一个SQL语句中的查询,应用程序会很容易受到SQL注入,例如下面的例子: $unsafe_variable = $_POST['user_input'];mysql_query("INSERT INTO table (column) VALUES ('" . $unsafe_variable . "')"); 这是因为用户可以输入类似VALUE”); DROP TA

PHP防止SQL注入的方法(1)

(1)mysql_real_escape_string – 转义 SQL 语句中使用的字符串中的特殊字符,并考虑到连接的当前字符集 使用方法如下: $sql = "select count(*) as ctr from users where username ='".mysql_real_escape_string($username)."' and password='". mysql_r

PHP7扩展开发之依赖其他扩展

前言 有的时候,我们的扩展要依赖其他扩展。比如,我们PHP的mysqli扩展就依赖mysqlnd扩展。这中情况下,我们怎么使用其他扩展呢?这个就是本文讲述的内容。 我们新建立一个扩展,名字叫 demo_dep , 依赖之前的say扩展。 在demo_dep扩展中,我们实现demo_say方法。这个方法调用say扩展的say方法。 代码 基础代码 确保say扩展的头文件正确安装到了php

Golang进程权限调度包runtime

关于 runtime 包几个方法: Gosched:让当前线程让出 cpu 以让其它线程运行,它不会挂起当前线程,因此当前线程未来会继续执行GOMAXPROCS:设置最大的可同时使用的 CPU 核数Goexit:退出当前 goroutine(但是defer语句会照常执行)NumGoroutine:返回正在执行和排队的任务总数GOOS:目标操作系统NumCPU:返回当前系统的 CPU 核数量 p