写一个符合我当前问题的标题有点困难…我有一个main()函数,它使用另一个包(database_sql)中的一个函数。这个函数初始化一个全局变量sql.DB*。初始化后,变量不是nil,但是对于其他函数,这个变量仍然是nil…让我们看看下面的代码!
main.go
package main
import (
"net/http"
db "./database_sql"
router "./router"
)
func main() {
db.Init_SQL();
router.Init_routes()
http.ListenAndServe(":8080", router.GetRouter())
}
db.go
package database_sql
import (
"fmt"
"database/sql"
_ "github.com/go-sql-driver/mysql"
)
var DB *sql.DB // global variable to share it between main and the HTTP handler
//var dbi *databaseinfos
func Init_SQL() {
dbi := databaseinfos{
user: "something",
password: "something",
name: "something",
address: "something",
port: "2323",
url: ""}
dbi.url = (dbi.user + ":" + dbi.password + "@tcp(" + dbi.address + ":" + dbi.port + ")/" + dbi.name)
db_init_var, err := sql.Open("mysql", dbi.url)
if err != nil {
fmt.Println("Error on initializing database connection: %s", err.Error())
}
err = db_init_var.Ping()
if err != nil {
fmt.Println("Error on opening database connection: %s", err.Error())
}
// Here, as you can test, my variable is initialized
if err == nil {
DB = db_init_var
if DB == nil {
fmt.Println("NIL DB !!")
}
if db_init_var == nil {
fmt.Println("NIL db_init_var !!")
}
fmt.Println(dbi.url)
}
}
现在,一切正常!我现在将测试超文本传输协议://xxxxxxx/用用户名/密码登录。一切正常,现在,是时候用上面声明的数据库变量在数据库中发出请求了。
请求用户.go
package database_sql
import (
"encoding/json"
"fmt"
"net/http"
m "models"
)
func CanLogin(user m.User) (bool, m.User, m.StatusBack) {
// Prepare statement for reading data
stmtOut, err := DB.Prepare("SELECT username, password, token FROM users WHERE username = ? AND password = ?")
if err != nil {
return false, user, m.StatusBack{ToString: err.Error(), IdStatus: http.StatusInternalServerError}
}
defer stmtOut.Close()
// Query the user
err = stmtOut.QueryRow(user.Username, user.Password).Scan(&user.Username, &user.Password, &user.UUID) // WHERE user is registered
if err != nil {
return false, user, m.StatusBack{ToString: "User not found.", IdStatus: http.StatusNotFound}
} else {
j, err := json.Marshal(user)
if err == nil {
return true, user, m.StatusBack{ToString: string(j), IdStatus: http.StatusOK}
} else {
return false, user, m.StatusBack{ToString: err.Error(), IdStatus: http.StatusInternalServerError}
}
}
}
然而,当我发出我的sql请求时,下面一行显示DB是nil。stmtOut,err := DB。Prepare("SELECT username,password,token FROM users WHERE username =?而密码=?”)
我做了很多测试,例如尝试用“=”而不是没有“:=”初始化它。我尝试了“如果DB == nil”语句,DB总是nil。为什么?我在初始化路由器之前初始化了我的数据库 var,所以从技术上讲,当我的路由器重定向到函数时,这个函数不会使用数据库 var (DB) nil,不是吗?
谢谢你的回答!
编辑2我编辑了你的答案,然后,我按照你的要求编辑了我的代码,并添加了一些内容。在init_SQL之后,DB不是零!
main.go
package main
import (
"fmt"
"net/http"
db "./database_sql"
router "./router"
)
func main() {
if db.Init_SQL() == true {
if db.DB == nil {
fmt.Println("db.DB is nil !")
}
if router.Init_routes() == true {
http.ListenAndServe(":8080", router.GetRouter())
}
}
}
所以,现在我的 2 个 init 函数在成功时返回 true,在失败时返回 false。我认为这可能是一个异步问题,我这样做是为了等待每个函数结束以继续我的“一步一步”
首先,输出这样的全球变量并不是偶像。复杂类型应该由运行时初始化(使用 main()
执行此操作),但从运行时全局保留以控制处置。
您还缺少数据库。关闭()。
我相信几年前第一次使用MySQL时,我也遇到过你的模式的同样问题。创建一个局部作用域指针,然后将它赋给一个全局变量的方式很奇怪。通常最好直接将其分配给全局var。但我认为核心问题是你需要把指针分配给指针。
我使用的模式是将可测试的数据库/sql *DB
保持在我初始化它的全局状态。当轮子运转时,为什么要重新发明轮子:
package main
import (
...
"database/sql"
"mypackage-that-inits-db/database"
)
var db *sql.DB
find main() {
var err error
db, err = database.Init_SQL(...params)
if err != nil {
...
}
defer db.Close()
InitUsers(db)
InitStats(db)
...
http.ListenAndServe(...)
// now inject db global dependency in other code
//
globalStats := NewStatsEngine(db)
globalStats.RecordUpdateToSomething(...)
users := getUsers(db)
... etc
}
这通常是您在其他代码中看到的模式。
请注意对 delay 和 Close 的控件,它属于调用方中的此处。
另请注意,通过创建一个模拟sql.DB
对象并将其注入NewstatsEngine、get用户等,您在其他包中的代码现在变得易于测试,而不必模拟全局变量、测试、拆除和再次设置以进行测试测试、测试、拆除等。此外,从Go 1.5开始,测试可以同时运行,因此如果您这样离开包,您甚至需要在全局数据库变量周围放置一个互斥锁。切换到IoC模式,如我在上面的代码中演示的依赖注入,可以更轻松地测试(和调试)您的代码。
问题内容: 我知道当我初始化一个char数组时: 要么 为什么不喜欢 初始化数组: 为什么它们不同?它是Java哲学的本质之一还是其背后的某些原因? 问题答案: 如果您曾经使用过 C ,那么答案就非常简单。在 C语言中 ,创建数组的方式是在堆栈上分配一个足以容纳元素数量的静态内存长度,并使用指针指向第一个元素-或堆上动态内存长度,然后用指针指向第一个元素。 在 C ++中 ,第二个版本已更改为
主要内容:变量初始化的标准格式,编译器推导类型的格式,短变量声明并初始化正如上一节《 Go语言变量声明》中提到的 Go语言在声明变量时,自动对变量对应的内存区域进行初始化操作。每个变量会初始化其类型的默认值,例如: 整型和浮点型变量的默认值为 0 和 0.0。 字符串变量的默认值为空字符串。 布尔型变量默认为 bool。 切片、函数、指针变量的默认为 nil。 当然,依然可以在变量声明时赋予变量一个初始值。 回顾C语言 在C语言中,变量在声明时,并不会对变量对应内存区
问题内容: 我有两个班,第一个是我的主班,第二个是我的编辑框架班。 我的第二个类(UpdateGUI)在其构造函数中提供oldName,并对其进行编辑,当我单击时,它将newName发送给我的第一个类。 我的第二堂课: 我的问题是,为什么newName为null? 更新: UpdateGUIDialog类: 输出: 我需要打印而不是null。 问题答案: Java对象有点像真实对象。并顾名思义:它
问题内容: 如何使用类型初始化变量? 因为我有错误 问题答案: 常量声明不能包含函数调用(某些例外,请参见下文),它们必须在编译时进行评估,而函数调用是在运行时进行的。 引用规格:常量: 常数值由符文,整数,浮点数,虚数或字符串文字表示,标识符表示常数,常数表达式,结果为常数的转换或某些内置结果的值函数,例如应用于任何值,或应用于某些表达式,以及应用于常量,以及应用于数字常量的复数。 并引用Spe
问题内容: 这行代码给出以下警告: 警告:变量sh只能在此位置为null。 并且,此代码给出以下警告: 警告:局部变量sh可能尚未初始化。 问题答案: 这是因为您需要初始化数组。试试这个: 如果不初始化,则会收到这些警告,如果运行它也会得到警告。
1、什么是初始化函数 初始化函数的意思是,当你创建一个实例的时候,这个函数就会被调用。 比如: 当代码在执行 a = ClassA() 的语句时,就自动调用了 __init__(self) 函数。 而这个 __init__(self) 函数就是初始化函数,也叫构造函数。 初始化函数的写法是固定的格式:中间是 init,意思是初始化,然后前后都要有【两个下划线】,然后 __init__() 的括号中