Go 框架中依赖注入的替代方案
在 Go 框架中,依赖注入 (DI) 是提供依赖的一种常见方法。但是,DI 可能会引入复杂性和额外代码。对于小型到中型的项目,DI 可能不必要。
以下是一些 DI 的替代方案:
- 构造函数注入:直接在对象的构造函数中传递依赖项,就像以下示例所示:
type Service struct {
db *sql.DB
}
func NewService(db *sql.DB) *Service {
return &Service{db}
}
- 局部作用域变量:在函数内声明并初始化依赖项,就像以下示例所示:
func HandleRequest(w http.ResponseWriter, r *http.Request) {
db := connectToDatabase()
service := NewService(db)
service.HandleRequest(w, r)
}
- 单例模式:为依赖项创建一个单例,并在需要时访问它,就像以下示例所示:
var db *sql.DB
func init() {
db = connectToDatabase()
}
func GetDB() *sql.DB {
return db
}
- 全局变量:虽然不推荐,但可以在全局作用域中定义依赖项变量,就像以下示例所示:
警告:全局变量可能导致难以调试和维护的耦合代码。
立即学习“go语言免费学习笔记(深入)”;
var globalDB *sql.DB
func init() {
globalDB = connectToDatabase()
}
实战案例
考虑一个处理 HTTP 请求的 simple Service:
type Service struct {
db *sql.DB
}
func (s *Service) HandleRequest(w http.ResponseWriter, r *http.Request) {
// 执行数据库操作
}
使用构造函数注入,我们可以:
package main
import (
"net/http"
"<a style='color:#f60; text-decoration:underline;' href="https://www.php.cn/zt/15841.html" target="_blank">git</a>hub.com/go-sql-driver/<a style='color:#f60; text-decoration:underline;' href="https://www.php.cn/zt/15713.html" target="_blank">mysql</a>"
)
type Service struct {
db *sql.DB
}
func (s *Service) HandleRequest(w http.ResponseWriter, r *http.Request) {
// 执行数据库操作
}
func main() {
db, err := sql.Open("mysql", "user:password@tcp(localhost:3306)/database")
if err != nil {
panic(err)
}
service := Service{db}
http.HandleFunc("/", service.HandleRequest)
http.ListenAndServe(":8080", nil)
}
或者,使用局部作用域变量:
package main
import (
"net/http"
"github.com/go-sql-driver/mysql"
)
func main() {
db, err := sql.Open("mysql", "user:password@tcp(localhost:3306)/database")
if err != nil {
panic(err)
}
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
service := Service{db}
service.HandleRequest(w, r)
})
http.ListenAndServe(":8080", nil)
}