2015-12-09 07:45:11 +00:00
|
|
|
### 2.3.3 new 函數
|
|
|
|
|
|
|
|
|
2015-12-18 06:49:31 +00:00
|
|
|
另一個創建變量的方法是用內建的 new 函數. 表達式 `new(T)` 創建一個T類型的匿名變量, 初始化爲T類型的零值, 返迴返迴變量地址, 返迴指針類型爲 `*T`.
|
2015-12-09 07:45:11 +00:00
|
|
|
|
|
|
|
```Go
|
2015-12-18 02:53:03 +00:00
|
|
|
p := new(int) // p, *int 類型, 指向匿名的 int 變量
|
2015-12-09 07:45:11 +00:00
|
|
|
fmt.Println(*p) // "0"
|
2015-12-18 02:53:03 +00:00
|
|
|
*p = 2 // 設置 int 匿名變量的值爲 2
|
2015-12-09 07:45:11 +00:00
|
|
|
fmt.Println(*p) // "2"
|
|
|
|
```
|
|
|
|
|
|
|
|
|
2015-12-18 06:49:31 +00:00
|
|
|
從 new 創建變量和普通聲明方式創建變量沒有什麽區别, 除了不需要聲明一個臨時變量的名字外, 我們還可以在表達式中使用 `new(T)`. 換言之, new 類似是一種語法醣, 而不是一個新的基礎概念.
|
2015-12-09 07:45:11 +00:00
|
|
|
|
2015-12-18 02:53:03 +00:00
|
|
|
下面的兩個 newInt 函數有着相同的行爲:
|
2015-12-09 07:45:11 +00:00
|
|
|
|
|
|
|
```Go
|
|
|
|
func newInt() *int { func newInt() *int {
|
|
|
|
return new(int) var dummy int
|
|
|
|
} return &dummy
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
2015-12-18 02:53:03 +00:00
|
|
|
每次調用 new 都是返迴一個新的變量的地址, 因此下面兩個地址是不同的:
|
2015-12-09 07:45:11 +00:00
|
|
|
|
|
|
|
```Go
|
|
|
|
p := new(int)
|
|
|
|
q := new(int)
|
|
|
|
fmt.Println(p == q) // "false"
|
|
|
|
```
|
|
|
|
|
2015-12-18 06:49:31 +00:00
|
|
|
當然也有特殊情況: 如果兩個類型都是空的, 也就是説類型的大小是0, 例如 `struct{}` 和 `[0]int`, 有可能有相同的地址(依賴具體的語言實現).
|
2015-12-09 07:45:11 +00:00
|
|
|
|
2015-12-18 06:49:31 +00:00
|
|
|
new 函數使用相對比較少, 因爲對應結構體來説, 可以直接用字面量語法創建新變量的方法更靈活 (§4.4.1).
|
2015-12-09 07:45:11 +00:00
|
|
|
|
2015-12-18 06:49:31 +00:00
|
|
|
由於 new 隻是一個預定義的函數, 它併不是一個關鍵字, 因此我們可以將 new 重新定義爲别的類型. 例如:
|
2015-12-09 07:45:11 +00:00
|
|
|
|
|
|
|
```Go
|
|
|
|
func delta(old, new int) int { return new - old }
|
|
|
|
```
|
|
|
|
|
2015-12-18 02:53:03 +00:00
|
|
|
因爲 new 被定義爲 int 類型的變量, 因此 delta 函數內部就無法在使用內置的 new 函數了.
|
2015-12-09 07:45:11 +00:00
|
|
|
|