Go语言函数参数传递中的指针变量陷阱:为何函数内修改外部变量无效?
本文分析一个Go语言程序中关于指针变量赋值的常见问题。程序尝试在函数内部修改指向数据库连接对象的指针变量,但主函数中该变量却未被修改。
以下代码片段展示了这个问题:
var db *sql.DB func main() { initDB(db) fmt.Println(db) // 打印 <nil> } func initDB(db *sql.DB) { var err error db, err = sql.Open("mysql", "root:123456@tcp(127.0.0.1:3306)/data") checkErr(err) db.SetMaxOpenConns(100) db.SetMaxIdleConns(100) db.SetConnMaxLifetime(time.Minute * 3) if err := db.Ping(); err != nil { checkErr(err) } fmt.Println(db) // 这里打印的是正确的值 } func checkErr(err error) { if err != nil { panic(err) } }
登录后复制
mn 函数打印的 db 仍然是 nil,而 initDB 函数内部已成功连接数据库。这是因为 Go 语言的函数参数传递机制是值传递。即使参数是指针类型,传递的也是指针值的副本。
立即学习“”;
在 initDB 函数中,db, err := sql.Open(…) 创建了一个新的 *sql.DB 对象,并将该对象赋值给 initDB 函数内部的局部变量 db。这并没有修改 main 函数中 db 变量的指向。initDB 函数结束后,其局部变量 db 被销毁,main 函数中的 db 仍然是 nil。
要正确修改 main 函数中的 db 变量,需要使用指针的指针,或者在 initDB 函数中使用指针接收器。 以下是如何使用指针的指针来解决这个问题:
var db *sql.DB func main() { initDB(&db) fmt.Println(db) // 现在将打印正确的值 } func initDB(db **sql.DB) { var err error *db, err = sql.Open("mysql", "root:123456@tcp(127.0.0.1:3306)/data") checkErr(err) (*db).SetMaxOpenConns(100) (*db).SetMaxIdleConns(100) (*db).SetConnMaxLifetime(time.Minute * 3) if err := (*db).Ping(); err != nil { checkErr(err) } fmt.Println(*db) // 这里打印的是正确的值 } func checkErr(err error) { if err != nil { panic(err) } }
登录后复制
通过传递 db 的地址 (&db),initDB 函数可以直接修改 main 函数中 db 变量的指向。
以上就是Go语言函数参数传递:指针变量在函数内修改后外部却不变?的详细内容,更多请关注php中文网其它相关文章!