詳解Go語(yǔ)言中new和make關(guān)鍵字的區(qū)別
本篇文章來(lái)介紹一道非常常見的面試題,到底有多常見呢?可能很多面試的開場(chǎng)白就是由此開始的。那就是 new 和 make 這兩個(gè)內(nèi)置函數(shù)的區(qū)別。
其實(shí)這個(gè)問(wèn)題本身并不復(fù)雜,簡(jiǎn)單來(lái)說(shuō)就是,new 只分配內(nèi)存,而 make 只能用于 slice、map 和 chan 的初始化,下面我們就來(lái)詳細(xì)介紹一下。
new
new 是一個(gè)內(nèi)置函數(shù),它會(huì)分配一段內(nèi)存,并返回指向該內(nèi)存的指針。
其函數(shù)簽名如下:
源碼
//?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 函數(shù)只接受一個(gè)參數(shù),這個(gè)參數(shù)是一個(gè)類型,并且返回一個(gè)指向該類型內(nèi)存地址的指針。
同時(shí) new 函數(shù)會(huì)把分配的內(nèi)存置為零,也就是類型的零值。
使用
使用 new 函數(shù)為變量分配內(nèi)存空間:
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
上面的代碼是等價(jià)的,new(int) 將分配的空間初始化為 int 的零值,也就是 0,并返回 int 的指針,這和直接聲明指針并初始化的效果是相同的。
當(dāng)然,new 函數(shù)不僅能夠?yàn)橄到y(tǒng)默認(rèn)的數(shù)據(jù)類型分配空間,自定義類型也可以使用 new 函數(shù)來(lái)分配空間,如下所示:
type?Student?struct?{
???name?string
???age?int
}
var?s?*Student
s?=?new(Student)?//分配空間
s.name?=?"zhangsan"
fmt.Println(s)
這就是 new 函數(shù),它返回的永遠(yuǎn)是類型的指針,指針指向分配類型的內(nèi)存地址。需要注意的是,new 函數(shù)只會(huì)分配內(nèi)存空間,但并不會(huì)初始化該內(nèi)存空間。
make
make 也是用于內(nèi)存分配的,但是和 new 不同,它只用于 slice、map 和 chan 的內(nèi)存創(chuàng)建,而且它返回的類型就是這三個(gè)類型本身,而不是他們的指針類型。因?yàn)檫@三種類型本身就是引用類型,所以就沒(méi)有必要返回他們的指針了。
其函數(shù)簽名如下:
源碼
//?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
通過(guò)上面的代碼可以看出 make 函數(shù)的 t 參數(shù)必須是 slice、map 和 chan 中的一個(gè),并且返回值也是類型本身。
使用
下面用 slice 來(lái)舉一個(gè)例子:
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 內(nèi)容被類型 int 的零值填充,如:[]int{0, 0, 0}。
map 和 chan 也是類似的,就不多說(shuō)了。
總結(jié)
通過(guò)以上分析,總結(jié)一下 new 和 make 主要區(qū)別如下:
- make 只能用來(lái)分配及初始化類型為 slice、map 和 chan 的數(shù)據(jù)。new 可以分配任意類型的數(shù)據(jù);
- new 分配返回的是指針,即類型
*Type。make 返回類型本身,即Type; - new 分配的空間被清零。make 分配空間后,會(huì)進(jìn)行初始化;
到此這篇關(guān)于詳解Go語(yǔ)言中new和make關(guān)鍵字的區(qū)別的文章就介紹到這了,更多相關(guān)Go語(yǔ)言 new make內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Golang設(shè)計(jì)模式之外觀模式的實(shí)現(xiàn)
這篇文章主要介紹了Golang設(shè)計(jì)模式之外觀模式的實(shí)現(xiàn),外觀模式是一種常用的設(shè)計(jì)模式之一,是一種結(jié)構(gòu)型設(shè)計(jì)模式,它提供了一個(gè)簡(jiǎn)單的接口來(lái)訪問(wèn)復(fù)雜系統(tǒng)的各種功能,從而降低了系統(tǒng)的復(fù)雜度,需要詳細(xì)了解可以參考下文2023-05-05
Golang中Bit數(shù)組的實(shí)現(xiàn)方式
這篇文章主要介紹了Golang中Bit數(shù)組的實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-04-04
詳解Golang如何實(shí)現(xiàn)支持隨機(jī)刪除元素的堆
堆是一種非常常用的數(shù)據(jù)結(jié)構(gòu),它能夠支持在O(1)的時(shí)間復(fù)雜度獲取到最大值(或最小值)。本文主要介紹了如何實(shí)現(xiàn)支持O(log(n))隨機(jī)刪除元素的堆,需要的可以參考一下2022-09-09

