Go make 和 new的區別
在Go語言中:
make 被用來分配引用類型的內存: map, slice, channel
new 被用來分配除了引用類型的所有其他類型的內存: int, string, array等
本文主要給大家介紹了Go語言中函數new與make的使用和區別,關於Go語言中new和make是內建的兩個函數,主要用來創建分配類型內存。在我們定義生成變量的時候,可能會覺得有點迷惑,其實他們的規則很簡單,下面我們就通過一些示例說明他們的區別和使用,話不多說了,來一起看看詳細的介紹吧。
變量的聲明
?1 2 |
var i int
var s string
|
變量的聲明我們可以通過var關鍵字,然後就可以在程序中使用。當我們不指定變量的默認值時,這些變量的默認值是他們的零值,比如int類型的零值是0,string類型的零值是"",引用類型的零值是nil。
對於例子中的兩種類型的聲明,我們可以直接使用,對其進行賦值輸出。但是如果我們換成引用類型呢?
?1 2 3 4 5 6 7 8 9 |
package main
import (
"fmt"
)
func main() {
var i *int
*i=10
fmt.Println(*i)
}
|
這個例子會打印出什麽?0還是10?。以上全錯,運行的時候會painc,原因如下:
?1 |
panic: runtime error: invalid memory address or nil pointer dereference
|
從這個提示中可以看出,對於引用類型的變量,我們不光要聲明它,還要為它分配內容空間,否則我們的值放在哪裏去呢?這就是上面錯誤提示的原因。
對於值類型的聲明不需要,是因為已經默認幫我們分配好了。
要分配內存,就引出來今天的new和make。
new
對於上面的問題我們如何解決呢?既然我們知道了沒有為其分配內存,那麽我們使用new分配一個吧。
?1 2 3 4 5 6 |
func main() {
var i *int
i= new (int)
*i=10
fmt.Println(*i)
}
|
現在再運行程序,完美PASS,打印10。現在讓我們看下new這個內置的函數。
?1 2 3 4 |
// 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
|
它只接受一個參數,這個參數是一個類型,分配好內存後,返回一個指向該類型內存地址的指針。同時請註意它同時把分配的內存置為零,也就是類型的零值。
我們的例子中,如果沒有*i=10,那麽打印的就是0。這裏體現不出來new函數這種內存置為零的好處,我們再看一個例子。
?1 2 3 4 5 6 7 8 9 10 11 12 |
func main() {
u:= new (user)
u.lock.Lock()
u.name = "張三"
u.lock.Unlock()
fmt.Println(u)
}
type user struct {
lock sync.Mutex
name string
age int
}
|
示例中的user類型中的lock字段我不用初始化,直接可以拿來用,不會有無效內存引用異常,因為它已經被零值了。
這就是new,它返回的永遠是類型的指針,指向分配類型的內存地址。
make
make也是用於內存分配的,但是和new不同,它只用於chan、map以及切片的內存創建,而且它返回的類型就是這三個類型本身,而不是他們的指針類型,因為這三種類型就是引用類型,所以就沒有必要返回他們的指針了。
註意,因為這三種類型是引用類型,所以必須得初始化,但是不是置為零值,這個和new是不一樣的。
?1 |
func make(t Type, size ...IntegerType) Type
|
從函數聲明中可以看到,返回的還是該類型。
二者異同
所以從這裏可以看的很明白了,二者都是內存的分配(堆上),但是make只用於slice、map以及channel的初始化(非零值);而new用於類型的內存分配,並且內存置為零。所以在我們編寫程序的時候,就可以根據自己的需要很好的選擇了。
make返回的還是這三個引用類型本身;而new返回的是指向類型的指針。
其實new不常用
所以有new這個內置函數,可以給我們分配一塊內存讓我們使用,但是現實的編碼中,它是不常用的。我們通常都是采用短語句聲明以及結構體的字面量達到我們的目的,比如:
?1 2 |
i:=0
u:=user{}
|
這樣更簡潔方便,而且不會涉及到指針這種比麻煩的操作。
make函數是無可替代的,我們在使用slice、map以及channel的時候,還是要使用make進行初始化,然後才才可以對他們進行操作。
Go make 和 new的區別