项目背景
设计思路
应用开发思路
IOC-golang 可以帮助开发者更清晰地“构造一个对象”
// +ioc:autowire=true
// +ioc:autowire:type=normal
// +ioc:autowire:paramType=Config
// +ioc:autowire:constructFunc=New
type RedisClient struct {
client *redis.Client
ServiceImpl1 Service `singleton:"main.ServiceImpl1"` // inject Service 's ServiceImpl1
}
type Config struct {
Address string
Password string
DB string
}
func (c *Config) New(impl *Impl) (*Impl, error) {
dbInt, err := strconv.Atoi(c.DB)
if err != nil {
return impl, err
}
client := redis.NewClient(&redis.Options{
Addr: c.Address,
Password: c.Password,
DB: dbInt,
})
_, err = client.Ping().Result()
if err != nil {
return impl, err
}
impl.client = client
return impl, nil
}
基于 IOC-golang 框架开发的,一个包含了 redis 客户端的结构 RedisClient,通过注解或标签指定了参数、初始化逻辑和依赖的对象。
IOC-golang 可以帮助开发者管理设计模型
// +ioc:autowire=true
// +ioc:autowire:type=singleton
type App struct {
ServiceImpl1 Service `singleton:"main.ServiceImpl1"` // inject Service 's ServiceImpl1 implementation
RedisClientPtr *RedisClient `normal:",address=localhost:6379&db=0"` // inject RedisClient struct pointer
}
基于IOC-golang框架开发的,一个使用单例模式的对象 APP,其依赖一个 Service 接口,该接口期望被main.ServiceImpl1单例模式结构注入;APP 还依赖一个*RedisClient 结构体指针,期望以多例模式注入,并传入了参数。参数也可以从配置文件的结构默认位置读取。
如果按照常规的开发方式,开发者需要额外手动读取参数并创建依赖的 RedisClient 对象,手动组装 APP对象,维护单例模型指针,提供单例模型的构造函数等,这些逻辑都被 IOC-golang 封装好了。
对象生命周期
上一节所说的是静态的编码过程,这一节我们来聊一聊,一个对象从静态的编码,到应用运行过程中被加载直到销毁的整个生命周期。
结构定义 结构提供者编写结构的字段与函数。 参数传入与依赖对象加载 对象创建的准备阶段,获取到依赖的参数,获取到所有依赖的下游对象。 对象创建与初始化 对象的组装过程,组装完成后执行必要的初始化逻辑。 对象使用 对象的函数被使用者调用 对象销毁 对象被销毁
结构开发者视角:(撰写产品说明书) 我负责定义结构的字段和函数 我负责明确依赖的下游对象、依赖的参数字段。 我负责明确参数应该从哪里加载,例如:从标签加载,从配置中某个位置加载,从 API 传入的参数中加载。 我负责明确结构的依赖注入模型,比如单例模型。 我负责定义对象的初始化逻辑 我负责定义对象的销毁逻辑
结构使用者视角:(阅读产品说明书,按照说明书的内容使用结构) 我负责明确要使用哪个结构。(找到对应的产品说明书) 我负责使用一种“产品说明书”中支持的自动装载模型,例如:单例模型。 我负责明确“产品说明书”中给定的结构依赖参数。 我负责使用一种“产品说明书”里支持的参数加载方式,来加载结构依赖参数,加载方式可能包括:从标签加载,从配置中某个位置加载,从 API 传入的参数中加载。 我负责使用一种对象获取方式,来获取对象实例,例如:通过 API 获取,通过标签注入获取。 我负责调用对象函数。 我负责触发对象销毁逻辑。
可扩展性
对象的可扩展性 对象的可扩展性,即针对确定的一个结构(非单例),你可以通过传入不同的参数来获取多个期望的对象。这个过程往往被结构使用者关注,他需要思考如何传入参数,获得对象,调用对象从而实现正确的业务逻辑。通过这一可扩展性,结构使用者可以扩展出多个对象实例。
结构的可扩展性 结构的可扩展性,即针对一个确定的自动装载模型,你可以通过定义自己的结构描述信息,将你的结构体注册在框架上,以供使用。这个过程是结构提供者关心的,他需要思考选用哪个自动装载模型,思考提供的结构的全部生命周期信息。通过这一可扩展性,结构提供者可以为框架注入多种多样的结构,这些结构都会按照被选择的自动装载模型执行加载逻辑。 框架提供了一些预置的结构,例如 redis 客户端、gorm客户端等等,开发者可以直接传入参数,注入或通过API获取,直接使用,但这些预置的结构一定无法覆盖业务需求的。开发者注册自己的任何结构到框架上,都是使用了结构的可扩展性,这些被注册的结构和框架提供的预置结构,都是同一层面的概念。
自动装载模型的可扩展性 自动装载模型描述了一类结构的装载方式,例如是否单例模型、参数从哪里加载、注入标签应该符合什么格式等等。这个过程也是结构提供者关心的。 框架提供了一些预置的自动装载模型,例如单例模型、多例模型、配置模型、rpc 模型等,开发者可以根据按照业务需要,将一个或多个结构注册在期望的自动装载模型上,当已有的自动装载模型不足以适配业务场景,开发者可以调用API进行定制化。
小结
IOC-golang 框架关注:
一个问题:一个对象如何被创建。 两个角度:结构开发者视角,结构使用者视角。 三个维度扩展:自动装载模型、结构、对象。
项目层级与功能
项目层级
IOC-golang 是一个语言绑定的框架,处于应用开发的最高层,直接由开发人员操作。一些开发常用的组件所处层级如下图所示。这也解释了,IOC-golang 可以称为服务于 Go 开发者的 ioc 框架,而不是 Spring 框架的 Go 语言实现。
主要功能
主要功能有二,即 IOC 和 AOP,分别代表了面向开发的“依赖注入”能力,和面向运维的 “代理结构 AOP 层”。
依赖注入能力:IOC 依赖注入能力 灵活的参数传递能力
本框架支持通过标签、API、配置三种方式传入依赖参数,并允许开发者扩展这一能力。 标签参数传递:传递 address, db 等结构所需参数
强大、易用、可扩展的依赖注入能力,是框架的核心功能。
// +ioc:autowire=true
// +ioc:autowire:type=singleton
type App struct {
ServiceStruct *ServiceStruct `singleton:""` // inject ServiceStruct struct pointer
}
// +ioc:autowire=true
// +ioc:autowire:type=singleton
type App struct {
ServiceImpl1 ServiceInterface `singleton:"main.ServiceImpl1"` // inject ServiceInterface 's ServiceImpl1 implementation
}
可以通过 API 的方式,获取任何已注册的结构体:
// 该函数由 iocli 工具自动生成
func GetApp() (*App, error) {
i, err := singleton.GetImpl(util.GetSDIDByStructPtr(new(App)), nil)
if err != nil {
return nil, err
}
impl := i.(*App)
return impl, nil
}
// +ioc:autowire=true
// +ioc:autowire:type=singleton
type App struct {
NormalDB3Redis normalRedis.ImplIOCInterface `normal:",address=127.0.0.1:6379&db=3"`
}
通过 API 进行参数传递:
func main() {
client := redis.NewClient(&redis.Options{
Addr: p.RedisAddr,
})
}
autowire:
normal:
:
:
param:
nacosclientparam:
clientconfig:
appKey: appKey
serverconfigs:
ipaddr: 127.0.0.1
port: 8848
结构代理层:AOP
// +ioc:autowire=true// +ioc:autowire:type=singleton
type Impl1 struct {
}
func (i *Impl1) Hello(req string) string {
return req
}
结构注册代码,会以结构描述符的形式,将结构生命周期注册到框架上。 // +ioc:autowire=true
// +ioc:autowire:type=singleton
type App struct {
ServiceImpl1 Service `singleton:"main.ServiceImpl1"` // inject Service 's ServiceImpl1 implementation
RedisClientPtr *RedisClient `normal:",address=localhost:6379&db=0"` // inject RedisClient struct pointer
}
1
结构专属接口,命名为 $(结构名)IOCInterface
// +ioc:autowire=true
// +ioc:autowire:type=singleton
type App struct {
ServiceImpl1 Service `singleton:"main.ServiceImpl1"` // inject Service 's ServiceImpl1 implementation
RedisClientPtr *RedisClient `normal:",address=localhost:6379&db=0"` // inject RedisClient struct pointer
}
2
结构代理层存根 框架会为所有结构提供代理层,在面向接口编程的场景下,所有请求都会经过代理层,扩展出强大的运维能力。
// +ioc:autowire=true
// +ioc:autowire:type=singleton
type App struct {
ServiceImpl1 Service `singleton:"main.ServiceImpl1"` // inject Service 's ServiceImpl1 implementation
RedisClientPtr *RedisClient `normal:",address=localhost:6379&db=0"` // inject RedisClient struct pointer
}
3结构获取API // +ioc:autowire=true
// +ioc:autowire:type=singleton
type App struct {
ServiceImpl1 Service `singleton:"main.ServiceImpl1"` // inject Service 's ServiceImpl1 implementation
RedisClientPtr *RedisClient `normal:",address=localhost:6379&db=0"` // inject RedisClient struct pointer
}
4
结构展示:可以展示所有加载到框架的结构、方法列表
接口参数动态监听:可以监听运行态的 go 进程,展示当前正在执行的函数、参数值、返回值。
链路追踪:可以获取一个请求的完整调用链路,获取分布式场景下的跨进程调用链路,从而分析性能瓶颈。
链路可视化、分布式应用拓扑展示 告警 故障注入 问题诊断 ...
丰富的组件
IOC-golang 目前已预置一系列开发常用sdk,供直接注入。覆盖数据库、缓存、中间件、RPC 等等领域。目前已支持的组件有:
Nacos Rocketmq Redis gRPC GORM Dubbo3 HttpServer
愿景
IOC-golang 是一个崭新的项目,从第一行代码的编写,到今天不过一个月的时间。
我们的愿景是让 IOC-golang 成为 Go 应用开发的首选框架,让 Go 开发人员更贴近业务逻辑,减少冗余代码,增加代码的易读性,让 Go 生态的 “面向对象编程” 进入自动挡时代。
如何贡献
项目初创,十分期待能与感兴趣的开发者共创一片天地。
项目github地址: github.com/alibaba/ioc-golang
项目文档:ioc-golang.github.io
项目示例:ioc-golang/example
欢迎加入钉钉群“IOC-golang 交流群”:44638289
往期推荐
想要了解Go更多内容,欢迎扫描下方👇 关注 公众号,回复关键词 [实战群] ,就有机会进群和我们进行交流~
分享、在看与点赞,至少我要拥有一个叭~
推荐站内搜索:最好用的开发软件、免费开源系统、渗透测试工具云盘下载、最新渗透测试资料、最新黑客工具下载……
还没有评论,来说两句吧...