網(wǎng)站首頁 編程語言 正文
區(qū)別:在go語言中,make和new都是內存的分配(堆上),但是make只用于slice、map以及channel的初始化(非零值);而new用于類型的內存分配,并且內存置為零。make返回的是引用類型本身;而new返回的是指向類型的指針。
本文操作環(huán)境:windows10系統(tǒng)、GO 1.11.2、thinkpad t480電腦。
Go語言中new和make都是用來內存分配的原語(allocation primitives)。簡單的說,new只分配內存,make用于slice,map,和channel的初始化。
new
new(T)函數(shù)是一個分配內存的內建函數(shù)。
我們都知道,對于一個已經(jīng)存在變量,可對其指針進行賦值。
示例
var p int var v *int v = &p *v = 11 fmt.Println(*v)
那么,如果不是已經(jīng)存在的變量會如何呢?能對其直接賦值嗎?
var v *int *v = 8 fmt.Println(*v)
結果會報如下錯誤
panic: runtime error: invalid memory address or nil pointer dereference
[signal 0xc0000005 code=0x1 addr=0x0 pc=0x48df66]
如何解決?通過Go提供了new來初始化一地址就可以解決。
var v *int v = new(int) *v = 8 fmt.Println(*v)
那么我們來分析一下
fmt.Println(*v) fmt.Println(v) //<nil> v = new(int) fmt.Println(*v)// fmt.Println(v)//0xc00004c088
我們可以看到初始化一個指針變量,其值為nil,nil的值是不能直接賦值的。通過new其返回一個指向新分配的類型為int的指針,指針值為0xc00004c088,這個指針指向的內容的值為零(zero value)。
同時,需要注意的是不同的指針類型零值是不同的。
示例
type Name struct { P string } var av *[5]int var iv *int var sv *string var tv *Name av = new([5]int) fmt.Println(*av) //[0 0 0 0 0 0] iv = new(int) fmt.Println(*iv) // 0 sv = new(string) fmt.Println(*sv) // tv = new(Name) fmt.Println(*tv) //{}
上面講了對普通類型new()處理過后是如何賦值的,這里再講一下對復合類型(數(shù)組,slice,map,channel等),new()處理過后,如何賦值。
數(shù)組示例
var a [5]int fmt.Printf("a: %p %#v \n", &a, a)//a: 0xc04200a180 [5]int{0, 0, 0, 0, 0} av := new([5]int) fmt.Printf("av: %p %#v \n", &av, av)//av: 0xc000074018 &[5]int{0, 0, 0, 0, 0} (*av)[1] = 8 fmt.Printf("av: %p %#v \n", &av, av)//av: 0xc000006028 &[5]int{0, 8, 0, 0, 0}
silce 示例
var a *[]int fmt.Printf("a: %p %#v \n", &a, a) //a: 0xc042004028 (*[]int)(nil) av := new([]int) fmt.Printf("av: %p %#v \n", &av, av) //av: 0xc000074018 &[]int(nil) (*av)[0] = 8 fmt.Printf("av: %p %#v \n", &av, av) //panic: runtime error: index out of range
map 示例
var m map[string]string fmt.Printf("m: %p %#v \n", &m, m)//m: 0xc042068018 map[string]string(nil) mv := new(map[string]string) fmt.Printf("mv: %p %#v \n", &mv, mv)//mv: 0xc000006028 &map[string]string(nil) (*mv)["a"] = "a" fmt.Printf("mv: %p %#v \n", &mv, mv)//這里會報錯panic: assignment to entry in nil map
channel示例
cv := new(chan string) fmt.Printf("cv: %p %#v \n", &cv, cv)//cv: 0xc000074018 (*chan string)(0xc000074020) //cv <- "good" //會報 invalid operation: cv <- "good" (send to non-chan type *chan string)
通過上面示例我們看到數(shù)組通過new處理,數(shù)組av初始化零值,數(shù)組雖然是復合類型,但不是引用類型,其他silce、map、channel類型也屬于引用類型,go會給引用類型初始化為nil,nil是不能直接賦值的。并且不能用new分配內存。無法直接賦值。那么用make函數(shù)處理會是怎么樣呢?
make
示例
av := make([]int, 5) fmt.Printf("av: %p %#v \n", &av, av) //av: 0xc000046400 []int{0, 0, 0, 0, 0} av[0] = 1 fmt.Printf("av: %p %#v \n", &av, av) //av: 0xc000046400 []int{1, 0, 0, 0, 0} mv := make(map[string]string) fmt.Printf("mv: %p %#v \n", &mv, mv) //mv: 0xc000074020 map[string]string{} mv["m"] = "m" fmt.Printf("mv: %p %#v \n", &mv, mv) //mv: 0xc000074020 map[string]string{"m":"m"} chv := make(chan string) fmt.Printf("chv: %p %#v \n", &chv, chv) //chv: 0xc000074028 (chan string)(0xc00003e060) go func(message string) { chv <- message // 存消息 }("Ping!") fmt.Println(<-chv) // 取消息 //"Ping!" close(chv)
make不僅可以開辟一個內存,還能給這個內存的類型初始化其零值。
它和new還能配合使用
示例
var mv *map[string]string fmt.Printf("mv: %p %#v \n", &mv, mv)//mv: 0xc042004028 (*map[string]string)(nil) mv = new(map[string]string) (*mv) = make(map[string]string) fmt.Printf("mv: %p %#v \n", &mv, mv)//mv: 0xc042004028 &map[string]string{"a":"a"}
通過new給指針變量mv分配了一個內存,并賦予其內存地址。Map是引用類型,其零值為nil,使用make初始化 map,然后變量就能使用*給指針變量mv賦值了。
小結:
- make和new都是golang用來分配內存的內建函數(shù),且在堆上分配內存,make 即分配內存,也初始化內存。new只是將內存清零,并沒有初始化內存。
- make返回的還是引用類型本身;而new返回的是指向類型的指針。
- make只能用來分配及初始化類型為slice,map,channel的數(shù)據(jù);new可以分配任意類型的數(shù)據(jù)。
原文鏈接:https://www.php.cn/be/go/465883.html
相關推薦
- 2023-02-03 C++中的HTTP協(xié)議問題_C 語言
- 2024-03-07 Mapper批量映射優(yōu)化
- 2022-07-18 SQL?Server使用T-SQL進階之公用表表達式(CTE)_MsSql
- 2022-11-17 WPF利用DrawingContext實現(xiàn)繪制溫度計_C#教程
- 2022-07-15 python中進程間通信及設置狀態(tài)量控制另一個進程_python
- 2022-08-13 beginInvoke加回調函數(shù)lamad
- 2022-09-08 pandas庫中to_datetime()方法的使用解析_python
- 2022-07-13 Linux下網(wǎng)絡配置和ifconfig命令的使用
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細win安裝深度學習環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結構-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支