[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