最近用go寫(xiě)程序時(shí)遇到一個(gè)問(wèn)題——求任意類(lèi)型切片的長(zhǎng)度。
作為一個(gè)初學(xué)者,剛剛學(xué)了接口和切片,知道了每個(gè)類(lèi)型都實(shí)現(xiàn)了一個(gè)空接口interface{},那么如果接口類(lèi)型作為函數(shù)的參數(shù),那它應(yīng)該是可以接收任意類(lèi)型的實(shí)參的
帶著這樣的想法就寫(xiě)出了下面的代碼:
func size(ins []interface{}) int {
return len(ins)
}
然后調(diào)用
a := []int{1, 2, 3, 4}
fmt.Println(size(a))
但編譯的時(shí)候報(bào)了以下錯(cuò)誤:
cannot use a (type []int) as type []interface {} in argument to size
從報(bào)錯(cuò)的信息來(lái)看,是go語(yǔ)言不支持將任意類(lèi)型的切片轉(zhuǎn)換為接口切片所導(dǎo)致的,為了確定是go語(yǔ)言本身不支持所導(dǎo)致以及探究不支持的原因,于是在網(wǎng)上查閱了一些資料,最權(quán)威的應(yīng)該是來(lái)自于go的官方文檔
這上邊的解釋是說(shuō),由于非接口類(lèi)型的切片與接口類(lèi)型的切片在內(nèi)存中的空間布局不一樣,如果要做這樣的隱式轉(zhuǎn)換,將會(huì)比較耗時(shí),因此go不支持此種轉(zhuǎn)換。
如果確實(shí)需要用到傳接口切片的情況,則需要由程序員自己來(lái)提前做轉(zhuǎn)換,在傳參的時(shí)候確保實(shí)參是接口切片類(lèi)型,這樣才能通過(guò)編譯,這也是官方推薦的做法
代碼如下所示:
// 獲得一個(gè)int類(lèi)型的切片
var dataSlice []int = foo()
// 創(chuàng)建接口類(lèi)型的切片
var interfaceSlice []interface{} = make([]interface{}, len(dataSlice))
// 依次轉(zhuǎn)換每個(gè)元素
for i, d := range dataSlice {
interfaceSlice[i] = d
}
// 調(diào)用上面的size方法
size(interfaceSlice)
如果按照上面的寫(xiě)法來(lái)傳參,那么求切片的長(zhǎng)度就顯得太費(fèi)勁了,還不如直接調(diào)用 len(dataSlice) 完事。
事情發(fā)展到這里,有點(diǎn)不甘心,于是繼續(xù)查資料,發(fā)現(xiàn)go語(yǔ)言的反射機(jī)制可以解決這個(gè)問(wèn)題。
首先將上面的size函數(shù)的參數(shù)改為接口類(lèi)型(a interface{}),然后在里面用reflect.TypeOf(a).Kind()來(lái)判斷實(shí)參的類(lèi)型,如果是切片類(lèi)型,則用reflect.ValueOf()來(lái)獲得該切片,最后返回切片的長(zhǎng)度
代碼如下所示:
func Size(a interface{}) int {
if reflect.TypeOf(a).Kind() != reflect.Slice {
return -1
}
ins := reflect.ValueOf(a)
return ins.Len()
}
補(bǔ)充:Go語(yǔ)言中切片的長(zhǎng)度與容量的變化
在學(xué)習(xí)go語(yǔ)言的切片信息時(shí),發(fā)現(xiàn)切片的容量變化非常讓人摸不著頭腦,為了更記憶深刻就寫(xiě)下了這篇,如有錯(cuò)誤之處,請(qǐng)大家指正
一、當(dāng)前切片的長(zhǎng)度與容量相等情況:
package main
import (
"fmt"
)
func main() {
numbers := []int{0,1,2}
printSlice(numbers)
//通過(guò)append給numbers增加信息,如果當(dāng)前切片的長(zhǎng)度與容量相等,增加信息的長(zhǎng)度小于等于原來(lái)的長(zhǎng)度,
那么切片的長(zhǎng)度變?yōu)橄嗉又?,容量變?yōu)樵瓉?lái)的2倍(圖片一的第二行結(jié)果)
numbers = append(numbers, 10,5,6)
printSlice(numbers)
//通過(guò)append給numbers增加信息,如果當(dāng)前切片A的長(zhǎng)度與容量相等,增加信息B的長(zhǎng)度大于切片A原來(lái)的長(zhǎng)度,
那么切片的長(zhǎng)度變?yōu)橄嗉又?,容量變?yōu)椋築長(zhǎng)度+A長(zhǎng)度+(B長(zhǎng)度-A長(zhǎng)度)%2(圖片一的第三行結(jié)果)
numbers = append(numbers, 12,13,15,16,17,18,19,20,21,22,23)
printSlice(numbers)
}
func printSlice(x []int){
fmt.Printf("len=%d cap=%d slice=%v\n",len(x),cap(x),x)
}
圖片一:
二、如果當(dāng)前切片的長(zhǎng)度小于容量情況:
1、增加信息的長(zhǎng)度與當(dāng)前長(zhǎng)度和小于等于容量
func main() {
/* 創(chuàng)建切片 */
numbers := []int{0,1,2}
printSlice(numbers)
numbers = append(numbers, 10,5)
printSlice(numbers)
//通過(guò)append給numbers增加信息,如果當(dāng)前切片的長(zhǎng)度小于容量,增加信息的長(zhǎng)度與當(dāng)前長(zhǎng)度和小于等于容量,
那么numbers的長(zhǎng)度變?yōu)橄嗉又?,容量不?圖片二的第三行結(jié)果)
numbers = append(numbers, 11)
printSlice(numbers)
}
圖片二:
2、增加信息B的長(zhǎng)度與當(dāng)前A的長(zhǎng)度大于A容量并且小于A容量的2倍
func main() {
/* 創(chuàng)建切片 */
numbers := []int{0,1,2}
printSlice(numbers)
numbers = append(numbers, 10,5)
printSlice(numbers)
//通過(guò)append給numbers增加信息,如果當(dāng)前切片A的長(zhǎng)度小于容量,增加信息B的長(zhǎng)度與當(dāng)前A的長(zhǎng)度大于A容量并且小于A容量的2倍,
那么numbers的長(zhǎng)度變?yōu)橄嗉又?,容量變?yōu)椋篈容量*2(圖片三的第三行結(jié)果)
numbers = append(numbers, 11,12)
printSlice(numbers)
}
圖片三:
3、增加信息B的長(zhǎng)度與當(dāng)前A的長(zhǎng)度大于A容量的2倍
func main() {
/* 創(chuàng)建切片 */
numbers := []int{0,1,2}
printSlice(numbers)
numbers = append(numbers, 10,5)
printSlice(numbers)
//通過(guò)append給numbers增加信息,如果當(dāng)前切片A的長(zhǎng)度小于容量,增加信息B的長(zhǎng)度與當(dāng)前A的長(zhǎng)度大于A容量的2倍,
那么numbers的長(zhǎng)度變?yōu)橄嗉又虲。容量變?yōu)椋築長(zhǎng)度+A長(zhǎng)度+(B長(zhǎng)度-A長(zhǎng)度)%2(圖片四的第三行結(jié)果)
numbers = append(numbers, 11,12,13,15,16,17,18,19,20)
printSlice(numbers)
}
圖片四:
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
您可能感興趣的文章:- 如何在Go中使用切片容量和長(zhǎng)度
- Go語(yǔ)言切片前或中間插入項(xiàng)與內(nèi)置copy()函數(shù)詳解
- golang中切片copy復(fù)制和等號(hào)復(fù)制的區(qū)別介紹
- go語(yǔ)言中切片與內(nèi)存復(fù)制 memcpy 的實(shí)現(xiàn)操作
- go語(yǔ)言中的二維切片賦值
- go語(yǔ)言中切片的長(zhǎng)度和容量的區(qū)別