[]int 能转换为 []interface 吗?
這個問題的答案是:不能。
如果你還想知道更多的信息,就往下看。^_^
有些時候我們希望有這樣的寫法:定義一個參數為 []interface 的函數,在程序運行的過程中,傳入 []int 或其他類型的 slice,以此來達到少寫一些代碼的目的。譬如下面這個弱智的求 slice 和的例子:
package?mainimport?"fmt"func?sliceSum(inters?[]interface{})?(res?interface{})?{nums?:=?inters.([]int)sum?:=?0for?_,?num?:=?range?nums?{sum?+=?num}return?sum }func?main()?{is?:=?[]int{7,?8,?9,?10}fmt.Println(sliceSum(is)) }為了把這個程序寫得更通用一點,參數和返回值都是用的 interface 類型。編譯,會報錯:
./inter.go:6:16:?invalid?type?assertion:?inters.([]int)?(non-interface?type?[]interface?{}?on?left) ./inter.go:19:22:?cannot?use?is?(type?[]int)?as?type?[]interface?{}?in?argument?to?sliceSum第一個錯:不能將左邊的 []interface{} 轉換成右邊的 []int,因為 []interface 本身并不是 interface 類型,所以不能進行斷言。
第二個錯:sliceSum 函數不能接受 []int 類型的參數,因為 []int 不是 []interface 類型。
先把程序改成正確的:
package?mainimport?"fmt"func?sliceSum(inters?[]interface{})?(res?interface{}){sum?:=?0for?_,?inter?:=?range?inters?{sum?+=?inter.(int)}return?sum }func?main()?{is?:=?[]int{7,?8,?9,?10}iis?:=?make([]interface{},?len(is))for?i?:=?0;?i?<?len(is);?i++?{iis[i]?=?is[i]}fmt.Println(sliceSum(iis)) }直接在循環的地方,對 inters 里的每個元素進行斷言后再累加。
再來研究下 Go 官方說的:[]int 和 []interface{} 內存模型不一樣是什么意思。
之前的 slice 文章講過,slice 底層有 3 個屬性:
sliceinterface 的文章講過,interface 底層有兩個屬性:
interface用 dlv 來調試,在關鍵地方打上斷點:
image知道了 slice 地址后,打印出該地址處的數據:
x?-fmt?hex?-len?24?0xc000055f30 int slice第一行即 slice 底層的數組地址,0x04, 0x04 分別指的是長度、容量。0x07、0x08、0x09、0x0a 則是數組的四個元素。
slice memory同樣的方法,來看看 interface slice 的內存布局:
interface slice其實也非常清楚,它的數據部分占 64 字節:因為一個 interface{} 占用 16 個字節,4 個元素所有是 64 個字節。
interface memory最后,總結一下:Go 官方規定,[]int 不能轉換成 []interface{},因為兩者是不同的類型,[]interface 不是 interface 類型,且兩者的內存布局并不相同。
解決辦法就是泛型。那泛型的原理是什么呢?又是怎么實現的呢?問就是不知道~????
注:本文內容主要來自于 Eli 的博客[1]。
參考資料
[1]
博客: https://eli.thegreenplace.net/2021/go-internals-invariance-and-memory-layout-of-slices/
總結
以上是生活随笔為你收集整理的[]int 能转换为 []interface 吗?的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 如何学习 Go 语言的核心知识
- 下一篇: Go服务迁到K8s后老抽风重启? 记一次