ThinkChat2.0新版上线,更智能更精彩,支持会话、画图、阅读、搜索等,送10W Token,即刻开启你的AI之旅 广告
[TOC] 了解下函数内部的值传递。这里面可以衍生出很多面试题。 ## 引入案例 首先来个常规的函数使用,并思考这里有几次变量赋值? ```go func foo(a int) { fmt.Printf("a: %v\n", a) } func main() { a := 10 foo(a) } ``` 答案是有两次赋值的。可以通过查看变量内存地址是否一致的。如果foo函数的变量a与main函数的变量a的地址是不一致的话,那说明有两次赋值的。请通过以下代码验证 ```go func foo(a int) { fmt.Printf("foo.a: %v, pointer: %p\n", a, &a) } func main() { a := 10 fmt.Printf("mian.a pointeris: %p\n", &a) foo(a) } // 运行结果: // mian.a pointeris: 0xc00012a000 // foo.a: 10, pointer: 0xc00012a008 ``` >[info] 总结: > - main函数,变量 a 定义并赋值操作。 > - 调用函数。即有一次赋值操作,**函数体内无需重新定义变量**。 ## 函数变量的修改 示例一 函数的形参是字符串,修改变量的值。是不影响实参的变量的。 ```go func bar(x int) { x = 100 // 变量赋值 fmt.Printf("bar.x: %v\n", x) } func main() { x := 10 bar(x) fmt.Printf("main.x: %v\n", x) } // 运行结果: // bar.x: 100 // main.x: 10 ``` 示例二 函数的形参是切片,修改函数的切片元素。是影响实参的变量。 ```go func testSlice(s []string) { fmt.Printf("test.s: %v, pointer: %p\n", s, s) s[1] = "shanghai" // 修改切片的元素 fmt.Printf("test.s: %v, pointer: %p\n", s, s) s = append(s, "guangdong") // 扩容切片 fmt.Printf("test.s: %v, pointer: %p\n", s, s) } func main() { s := []string{"beijing", "guangdong", "hunan"} fmt.Printf("main.s: %v, pointer: %p\n", s, s) testSlice(s) fmt.Printf("main.s: %v, pointer: %p\n", s, s) } // 运行结果: // main.s: [beijing guangdong hunan], pointer: 0xc0000a0150 // test.s: [beijing guangdong hunan], pointer: 0xc0000a0150 // test.s: [beijing shanghai hunan], pointer: 0xc0000a0150 // test.s: [beijing shanghai hunan guangdong], pointer: 0xc0000b6120 // main.s: [beijing shanghai hunan], pointer: 0xc0000a0150 ``` >[info] 总结: > 函数形参类型决定函数之间修改变量是否有连带关系。 ## 函数变量修改【变种】 上面的段落可知,函数之间的变量是否相互独立的。取决于函数形参的数据类型。如果变量是值拷贝,两个函数变量互不相关。如果变量是引用拷贝,两个函数变量修改会有所不同。 现在需要实现一个函数形参类型是值拷贝(string, int, boot, float32/float64...)修改变量的值,另一个函数也跟着改变。 ```go func setAge(age *int) { fmt.Printf("setAge.age pointer is: %v, value: %d\n", age, *age) *age++ fmt.Printf("setAge.age: %v\n", *age) } func main() { age := 18 fmt.Printf("main.age pointer is: %p, value: %d\n", &age, age) setAge(&age) fmt.Printf("main.age: %v\n", age) } // 运行结果: // main.age pointer is: 0xc0000ba000, value: 18 // setAge.age pointer is: 0xc0000ba000, value: 18 // setAge.age: 19 // main.age: 19 ```