Go语言函数参数传递:为什么指针变量在函数内修改后外部却不变?

Go语言函数参数传递:为什么指针变量在函数内修改后外部却不变?

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)     } }

main 函数打印的 db 仍然是 nil,而 initDB 函数内部已成功连接数据库。这是因为 Go 语言的函数参数传递机制是值传递。即使参数是指针类型,传递的也是指针值的副本。

立即学习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语言函数参数传递:

© 版权声明
THE END
喜欢就支持一下吧
点赞9 分享