原文連結: Go 語言 new 和 make 關鍵字的區別
本篇文章來介紹一道非常常見的面試題,到底有多常見呢?可能很多面試的開場白就是由此開始的。那就是 new 和 make 這兩個內建函式的區別。
其實這個問題本身並不複雜,簡單來說就是,new 只分配記憶體,而 make 只能用於 slice、map 和 chan 的初始化,下面我們就來詳細介紹一下。
new
new 是一個內建函式,它會分配一段記憶體,並返回指向該記憶體的指標。
其函式簽名如下:
原始碼
// The new built-in function allocates memory. The first argument is a type,
// not a value, and the value returned is a pointer to a newly
// allocated zero value of that type.
func new(Type) *Type
從上面的程式碼可以看出,new 函式只接受一個引數,這個引數是一個型別,並且返回一個指向該型別記憶體地址的指標。
同時 new 函式會把分配的記憶體置為零,也就是型別的零值。
使用
使用 new 函式為變數分配記憶體空間:
p1 := new(int)
fmt.Printf("p1 --> %#v \n ", p1) //(*int)(0xc42000e250)
fmt.Printf("p1 point to --> %#v \n ", *p1) //0
var p2 *int
i := 0
p2 = &i
fmt.Printf("p2 --> %#v \n ", p2) //(*int)(0xc42000e278)
fmt.Printf("p2 point to --> %#v \n ", *p2) //0
上面的程式碼是等價的,new(int)
將分配的空間初始化為 int 的零值,也就是 0,並返回 int 的指標,這和直接宣告指標並初始化的效果是相同的。
當然,new 函式不僅能夠為系統預設的資料型別分配空間,自定義型別也可以使用 new 函式來分配空間,如下所示:
type Student struct {
name string
age int
}
var s *Student
s = new(Student) //分配空間
s.name = "zhangsan"
fmt.Println(s)
這就是 new 函式,它返回的永遠是型別的指標,指標指向分配型別的記憶體地址。需要注意的是,new 函式只會分配記憶體空間,但並不會初始化該記憶體空間。
make
make 也是用於記憶體分配的,但是和 new 不同,它只用於 slice、map 和 chan 的記憶體建立,而且它返回的型別就是這三個型別本身,而不是他們的指標型別。因為這三種型別本身就是引用型別,所以就沒有必要返回他們的指標了。
其函式簽名如下:
原始碼
// The make built-in function allocates and initializes an object of type
// slice, map, or chan (only). Like new, the first argument is a type, not a
// value. Unlike new, make's return type is the same as the type of its
// argument, not a pointer to it. The specification of the result depends on
// the type:
// Slice: The size specifies the length. The capacity of the slice is
// equal to its length. A second integer argument may be provided to
// specify a different capacity; it must be no smaller than the
// length, so make([]int, 0, 10) allocates a slice of length 0 and
// capacity 10.
// Map: An empty map is allocated with enough space to hold the
// specified number of elements. The size may be omitted, in which case
// a small starting size is allocated.
// Channel: The channel's buffer is initialized with the specified
// buffer capacity. If zero, or the size is omitted, the channel is
// unbuffered.
func make(t Type, size ...IntegerType) Type
透過上面的程式碼可以看出 make 函式的 t
引數必須是 slice、map 和 chan 中的一個,並且返回值也是型別本身。
使用
下面用 slice 來舉一個例子:
var s1 []int
if s1 == nil {
fmt.Printf("s1 is nil --> %#v \n ", s1) // []int(nil)
}
s2 := make([]int, 3)
if s2 == nil {
fmt.Printf("s2 is nil --> %#v \n ", s2)
} else {
fmt.Printf("s2 is not nill --> %#v \n ", s2)// []int{0, 0, 0}
}
slice 的零值是 nil
,但使用 make 初始化之後,slice 內容被型別 int 的零值填充,如:[]int{0, 0, 0}
。
map 和 chan 也是類似的,就不多說了。
總結
透過以上分析,總結一下 new 和 make 主要區別如下:
- make 只能用來分配及初始化型別為 slice、map 和 chan 的資料。new 可以分配任意型別的資料;
- new 分配返回的是指標,即型別
*Type
。make 返回型別本身,即Type
; - new 分配的空間被清零。make 分配空間後,會進行初始化;
以上就是本文的全部內容,如果覺得還不錯的話歡迎點贊,轉發和關注,感謝支援。
參考文章:
- https://go.dev/doc/effective_go#allocation_new
- http://c.biancheng.net/view/5722.html
- https://sanyuesha.com/2017/07/26/go-make-and-new/
推薦閱讀:
- [為什麼 Go 不支援 []T 轉換為 []interface](https://mp.weixin.qq.com/s/cwDEgnicK4jkuNpzulU2bw)
- 為什麼 Go 語言 struct 要使用 tags