相關問題在這里https://stackoverflow.com/a/12965872/6421681。在 go 中,您可以執行以下操作:func numsInFactorial(n int) (nums []int) { // `nums := make([]int)` is not needed for i := 1; i <= n; i++ { nums = append(nums, i) } return}但是,以下內容不起作用:func mapWithOneKeyAndValue(k int, v int) (m map[int]int) { m[k] = v return}拋出一個錯誤:panic: assignment to entry in nil map相反,您必須:func mapWithOneKeyAndValue(k int, v int) map[int]int { m := make(map[int]int) m[k] = v return}我找不到此行為的文檔。我已經通讀了 effective go 的所有內容,那里也沒有提到它。我知道命名的返回值是定義的(即內存已分配;接近所做的new)但未初始化(因此make不會復制行為)。經過一些實驗,我相信這種行為可以簡化為理解以下代碼的行為:func main() { var s []int // len and cap are both 0 var m map[int]int fmt.Println(s) // works... prints an empty slice fmt.Println(m) // works... prints an empty map s = append(s, 10) // returns a new slice, so underlying array gets allocated fmt.Println(s) // works... prints [10] m[10] = 10 // program crashes, with "assignment to entry in nil map" fmt.Println(m)}這個問題似乎append很可能調用make并分配一個新的切片來檢測容量s是0. 但是,map永遠不會得到顯式初始化。這個 SO 問題的原因有兩個。首先,我想記錄下 SO 上的行為。slice其次,為什么該語言允許和的非初始化定義map?根據我到目前為止的經驗,它似乎是一種實用的語言(即未使用的變量導致編譯失敗,gofmt 強制正確格式化),因此阻止代碼編譯是有意義的。
2 回答

繁華開滿天機
TA貢獻1816條經驗 獲得超4個贊
嘗試按索引分配 nil 切片 - 你會得到“恐慌:運行時錯誤:索引超出范圍”
append 函數與 nil 一起工作的唯一原因是 append 函數可以對給定的切片進行重新分配。例如,如果您嘗試將第 6 個元素附加到當前容量為 5 的 5 個元素的切片,它將創建具有新容量的新數組,從舊數組復制所有信息,并交換給定切片中的數據數組指針。在我的理解中,它只是動態數組的 golang 實現。
因此,nil slice 只是容量不足的 slice 的特例,因此它會在任何追加操作時重新分配。

慕斯王
TA貢獻1864條經驗 獲得超2個贊
一個 nil 映射在讀取時表現得像一個空映射,但嘗試寫入一個 nil 映射會導致運行時恐慌;不要那樣做。要初始化地圖,請使用內置的 make 函數
似乎 nil 映射被認為是有效的空映射,這就是他們不為其自動分配內存的原因。
- 2 回答
- 0 關注
- 164 瀏覽
添加回答
舉報
0/150
提交
取消