go语言依赖注入是什么

go语言中,依赖注入(DI)是一种解耦组件之间依赖关系的设计模式;在需要的时候,不同组件之间可以通过一个统一的界面获取其它组件中的对象和状态。依赖注入的好处是解耦;而解耦又能带来更多的好处:代码扩展性增强,代码的可维护性增强,更容易进行单元测试等等。

go语言依赖注入是什么

本教程操作环境:windows7系统、GO 1.18版本、Dell G3电脑。

依赖注入是什么?

第一次听到这个词的时候我是一脸懵逼的,很拗口有没有,可能很多学过spring的同学觉得这是很基础很好理解的知识,但因为我之前没学过Java和spring,所以第一次接触这个词的时候是很懵的。

依赖注入,英文名dependency injection,简称DI。依赖两个字很好理解,在软件设计上,从架构模块到函数方法都存在大大小小的依赖关系。

比如说在new A 之前需要先new B ,A依赖于B,这时候我们就可以说B是A的依赖,A控制B,AB之间存在着耦合的关系,而代码设计思想是最好可以做到松耦合。如果某一天B需要改造那么A也需要跟着改造。这是一个依赖你可以觉得没问题,但如果是A->B->C->D->E->F之间存在一连串的依赖关系,那么改造起来就会十分麻烦。

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

go语言依赖注入是什么

这个时候就需要一种东西来解开他们之间的强耦合,怎么解耦呢,只能借助第三方力量了,我们把A对B的控制权交给第三方,这种思想就称为控制反转(IOC Inversion Of Control),这个第三方称为IOC容器。而IOC容器要做的事情就是new一个B出来,然后把这个B的实例注入到A里面去,然后A就可以正常的使用基于B的方法了,这个过程被称为依赖项注入,而基于IOC的这种方法就叫做依赖注入。

简单来说,依赖注入(DI)是一种解耦组件之间依赖关系的设计模式。在需要的时候,不同组件之间可以通过一个统一的界面获取其它组件中的对象和状态。Go语言的接口设计,避免了很多需要使用第三方依赖注入框架的情况(比如Java,等等)。我们的注入方案只提供非常少的类似Dager或Guice中的注入方案,而专注于尽量避免手动去配置对象和组件之间的依赖关系。

依赖注入的好处

明白了依赖注入的思想,应该也就明白了其带来的最大好处——解耦。

而解耦又能带来更多的好处:代码扩展性增强,代码的可维护性增强,更容易进行单元测试等等。

那么依赖注入如何实现呢?

Java中有以下几种方式:

setter方法注入:实现特定属性的public set方法,来让外部容器调用传入所依赖类型的对象。

基于接口的注入:实现特定接口以供外部容器注入所依赖类型的对象。

基于构造函数的注入:实现特定参数的构造函数,在新建对象时传入所依赖类型的对象。

基于注解的注入:在代码里加上特定的关键字实现注入。

注解是最常见的方式,它像注释一样不被当做代码来执行,而是专门供别人阅读。但注释的读者完全是人类,而注解的主要读者除了人类之外还有框架或预编译器。

Go依赖注入-wire

wire就是一种基于注解的依赖注入方式。wire是 Google 开源的一个依赖注入工具,我们只需要在一个特殊的go文件中告诉wire类型之间的依赖关系,它会自动帮我们生成代码,帮助我们创建指定类型的对象,并组装它的依赖。

wire有两个基础概念,Provider(构造器)和Injector(注入器)。

通过提供provider函数,让wire知道如何产生这些依赖对象。wire根据我们定义的injector函数签名,生成完整的injector函数,injector函数是最终我们需要的函数,它将按依赖顺序调用provider。

wire的要求很简单,新建一个wire.go文件(文件名可以随意),创建我们的初始化函数。比如,我们要创建并初始化一个Mission对象,我们就可以这样:

//+build wireinjectpackage mainimport "github.com/google/wire"func InitMission(name string) Mission {  wire.Build(NewMonster, NewPlayer, NewMission)  return Mission{}}

登录后复制

可以看到第一行的注解:+build wireinject,表示这是一个注入器。+build其实是 Go 语言的一个特性。类似 C/C++ 的条件编译,在执行go build时可传入一些选项,根据这个选项决定某些文件是否编译。wire工具只会处理有wireinject的文件,所以我们的wire.go文件要加上这个。

在函数中,我们调用wire.Build()将创建Mission所依赖的类型的构造器传进去。例如,需要调用NewMission()创建Mission类型,NewMission()接受两个参数一个Monster类型,一个Player类型。Monster类型对象需要调用NewMonster()创建,Player类型对象需要调用NewPlayer()创建。所以NewMonster()和NewPlayer()我们也需要传给wire。

写完wire.go文件之后执行wire命令,就会自动生成一个wire_gen.go文件。

// Code generated by Wire. DO NOT EDIT.//go:generate wire//+build !wireinjectpackage main// Injectors from wire.go:func InitMission(name string) Mission {  player := NewPlayer(name)  monster := NewMonster()  mission := NewMission(player, monster)  return mission}

登录后复制

可以看到wire自动帮我们生成了InitMission方法,此方法中依次初始化了player,monster和mission。之后在我们的main函数中就只需调用这个InitMission即可。

func main() {  mission := InitMission("dj")  mission.Start()}

登录后复制

而在没用依赖注入之前,我们的代码是这样的:

func main() {  monster := NewMonster()  player := NewPlayer("dj")  mission := NewMission(player, monster)  mission.Start()}

登录后复制

是不是简洁了很多。这里只有三个对象的初始化,如果是更多可能才会意识到依赖注入的好处。

比如:

wire.go文件:// +build wireinject// The build tag makes sure the stub is not built in the final build.package diimport ("github.com/google/wire")//go:generate kratos t wirefunc InitApp() (*App, func(), error) {panic(wire.Build(dao.Provider, service.Provider, http.New, grpc.New, NewApp))}实现文件://daovar Provider = wire.NewSet(New, NewDB, NewRedis)//servicevar Provider = wire.NewSet(New, wire.Bind(new(pb.Server), new(*Service)))生成的wire_gen.go 文件:func InitApp() (*App, func(), error) {redis, cleanup, err := dao.NewRedis()if err != nil {return nil, nil, err}db, cleanup2, err := dao.NewDB()if err != nil {cleanup()return nil, nil, err}daoDao, cleanup3, err := dao.New(redis, db)if err != nil {cleanup2()cleanup()return nil, nil, err}serviceService, cleanup4, err := service.New(daoDao)if err != nil {cleanup3()cleanup2()cleanup()return nil, nil, err}engine, err := http.New(serviceService)if err != nil {cleanup4()cleanup3()cleanup2()cleanup()return nil, nil, err}server, err := grpc.New(serviceService)if err != nil {cleanup4()cleanup3()cleanup2()cleanup()return nil, nil, err}app, cleanup5, err := NewApp(serviceService, engine, server)if err != nil {cleanup4()cleanup3()cleanup2()cleanup()return nil, nil, err}return app, func() {cleanup5()cleanup4()cleanup3()cleanup2()cleanup()}, nil}

登录后复制

所以,依赖注入到底是什么?

封装解耦罢了。

【相关推荐:Go视频教程、编程教学】

以上就是go语言依赖注入是什么的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2414698.html

(0)
上一篇 2025年3月3日 00:44:14
下一篇 2025年3月3日 00:44:36

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • 如何查看Debian上的Golang日志

    本文介绍几种在Debian系统上查看Go语言应用日志的方法: 方法一:利用journalctl命令 如果你的Go应用以systemd服务的形式运行,可以使用journalctl命令查看其日志。 假设你的服务名为my-go-app,则使用以下…

    2025年4月1日
    000
  • Golang日志中敏感信息如何脱敏处理

    保障Golang应用日志安全,避免敏感信息泄露至关重要。本文介绍几种常见的Golang日志脱敏方法: 方法一:正则表达式替换 利用正则表达式匹配并替换敏感信息,例如邮箱和密码。 package mainimport ( “fmt” “log…

    2025年4月1日
    000
  • Golang日志如何高效分析

    高效处理Go语言日志,关键在于优化读取、解析和处理流程。以下策略能显著提升性能: 一、优化日志读取: 采用bufio包的缓冲读取方式,避免一次性加载所有数据到内存,显著提升大文件读取速度。合理设置缓冲区大小至关重要。 二、并发处理: 立即学…

    2025年4月1日
    000
  • Linux Golang日志如何优化

    在linux环境下使用golang进行日志优化,可以采取以下几种策略: 选择高效的日志库 zap:由Uber开源的高性能日志库,支持多种日志级别和输出方式,包括console、json、file等。zap使用Go语言本身的特性,如指针和结构…

    编程技术 2025年4月1日
    000
  • 如何通过Golang日志定位问题

    在golang中,日志是定位问题的关键工具之一。以下是一些建议,帮助您通过golang日志定位问题: 使用标准库log包:Golang的标准库log包提供了基本的日志功能,包括时间戳、日志级别和消息。您可以使用log.Println()、l…

    编程技术 2025年4月1日
    000
  • Golang日志在Debian如何分类

    本文介绍在Debian系统下有效管理和分类Golang应用日志的方法。 我们将探讨几种常见的技术和工具,帮助您分析和监控应用运行状况。 Golang日志库的选择 Golang标准库提供基础的日志功能,可通过log.SetFlags()函数自…

    2025年4月1日
    000
  • LiteIDE中如何快速查找和定位Go语言函数?

    高效使用LiteIDE查找和定位Go语言函数 在Go语言开发中,快速定位函数至关重要。本文介绍如何在LiteIDE中高效查找和使用Go语言函数,提升开发效率。 LiteIDE没有独立的函数列表窗口,但其代码补全和跳转功能可实现类似效果。 在…

    2025年3月31日
    100
  • Go语言Gorm框架下如何精准控制MySQL特定表的Binlog记录?

    使用Go语言Gorm框架操作MySQL数据库时,如何精确控制特定表的Binlog记录?本文将解决一个常见问题:即使设置了会话级别的sql_log_bin变量,特定表的操作仍然写入Binlog日志。 问题描述:用户尝试在操作表A之前设置sql…

    2025年3月31日
    100
  • Go语言中如何利用类型断言实现接口的静态检查?

    go语言类型断言:巧妙实现接口静态检查 本文将深入探讨Go语言中一种利用类型断言进行接口静态检查的技巧,并解释代码var _ cmder = (*deployCmd)(nil) 的作用。 这种技巧常用于Cobra等命令行框架中。 假设我们有…

    2025年3月31日
    100
  • Go语言中如何将十六进制字符串转换为uint8数组?

    Go语言中高效转换十六进制字符串为uint8数组 在Go语言开发中,经常需要处理十六进制字符串到uint8数组的转换。本文将探讨两种高效的转换方法,解决直接使用[]byte(string)转换失败的问题。 场景:开发者需要将一系列十六进制字…

    2025年3月31日
    100

发表回复

登录后才能评论