我有以下文件结构:
车型/ db.go
type DB struct {
*sql.DB
}
var db *DB
func init() {
dbinfo := fmt.Sprintf("user=%s password=%s dbname=%s sslmode=disable",DB_USER,DB_PASSWORD,DB_NAME)
db,err := NewDB(dbinfo)
checkErr(err)
rows,err := db.Query("SELECT * FROM profile")
checkErr(err)
fmt.Println(rows)
}
func NewDB(dataSourceName string) (*DB,error) {
db,err := sql.Open("postgres",dataSourceName)
if err != nil {
return nil,err
}
if err = db.Ping(); err != nil {
return nil,err
}
return &DB{db},nil
}
车型/ db_util.go
func (p *Profile) InsertProfile() {
if db != nil {
_,err := db.Exec(...)
checkErr(err)
} else {
fmt.Println("DB object is NULL")
}
}
当我尝试在InsertProfile函数中访问db时,它表示NULL ptr异常。如何在db_utils.go中访问db?
我不想利用db(因为它可以访问所有的包)。
我正在从init()中得到从数据库返回的QUERY。
编辑:问题是您使用 Short variable declaration:=,并且您刚刚将创建的* DB值存储在局部变量中,而不是全局变量中。这一行:
db,err := NewDB(dbinfo)
创建2个局部变量:db和err,并且此本地数据库与全局db变量无关。您的全局变量将保持为零。您必须将创建的* DB分配给全局变量。不要使用简短的变量声明,而是简单的assignment,例如:
var err error
db,err = NewDB(dbinfo)
if err != nil {
log.Fatal(err)
}
原来的答案如下。
它是一个指针类型,您必须在使用它之前初始化它。指针类型的零值为零。
你不必输出它(这是用大写字母开头的)。请注意,只要它们是同一个软件包的一部分,您就拥有多个文件,就可以访问彼此定义的标识符。
一个很好的解决方案是在自动调用的包init()函数中执行。
sql.Open()DB.Ping()
例如:
var db *sql.DB
func init() {
var err error
db,err = sql.Open("yourdrivername","somesource")
if err != nil {
log.Fatal(err)
}
if err = db.Ping(); err != nil {
log.Fatal(err)
}
}