[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
```
- Golang简介
- 开发环境
- Golang安装
- 编辑器及快捷键
- vscode插件
- 第一个程序
- 基础数据类型
- 变量及匿名变量
- 常量与iota
- 整型与浮点型
- 复数与布尔值
- 字符串
- 运算符
- 算术运算符
- 关系运算符
- 逻辑运算符
- 位运算符
- 赋值运算符
- 流程控制语句
- 获取用户输入
- if分支语句
- for循环语句
- switch语句
- break_continue_goto语法
- 高阶数据类型
- pointer指针
- array数组
- slice切片
- slice切片扩展
- map映射
- 函数
- 函数定义和调用
- 函数参数
- 函数返回值
- 作用域
- 函数形参传递
- 匿名函数
- 高阶函数
- 闭包
- defer语句
- 内置函数
- fmt
- strconv
- strings
- time
- os
- io
- 文件操作
- 编码
- 字符与字节
- 字符串
- 读写文件
- 结构体
- 类型别名和自定义类型
- 结构体声明
- 结构体实例化
- 模拟构造函数
- 方法接收器
- 匿名字段
- 嵌套与继承
- 序列化
- 接口
- 接口类型
- 值接收者和指针接收者
- 类型与接口对应关系
- 空接口
- 接口值
- 类型断言
- 并发编程
- 基本概念
- goroutine
- channel
- select
- 并发安全
- 练习题
- 第三方库
- Survey
- cobra