1. 程式人生 > >深入講解Go語言中函式new與make的使用和區別

深入講解Go語言中函式new與make的使用和區別

深入講解Go語言中函式new與make的使用和區別

 

 

大家都知道Go語言中的函式new與函式make一直是新手比較容易混淆的東西,看著相似,但其實不同,不過解釋兩者之間的不同也非常容易,下面這篇文章主要給大家介紹了關於Go語言中函式new與make區別的相關資料,需要的朋友可以參考下。

前言

本文主要給大家介紹了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進行初始化,然後才才可以對他們進行操作。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對指令碼之家的支援。

您可能感興趣的文章:

原文連結:http://www.flysnow.org/2017/10/23/go-new-vs-make.html