最近用go寫程序時遇到一個問題——求任意類型切片的長度。
作為一個初學者,剛剛學了接口和切片,知道了每個類型都實現(xiàn)了一個空接口interface{},那么如果接口類型作為函數(shù)的參數(shù),那它應該是可以接收任意類型的實參的
帶著這樣的想法就寫出了下面的代碼:
func size(ins []interface{}) int {
return len(ins)
}
然后調用
a := []int{1, 2, 3, 4}
fmt.Println(size(a))
但編譯的時候報了以下錯誤:
cannot use a (type []int) as type []interface {} in argument to size
從報錯的信息來看,是go語言不支持將任意類型的切片轉換為接口切片所導致的,為了確定是go語言本身不支持所導致以及探究不支持的原因,于是在網上查閱了一些資料,最權威的應該是來自于go的官方文檔
這上邊的解釋是說,由于非接口類型的切片與接口類型的切片在內存中的空間布局不一樣,如果要做這樣的隱式轉換,將會比較耗時,因此go不支持此種轉換。
如果確實需要用到傳接口切片的情況,則需要由程序員自己來提前做轉換,在傳參的時候確保實參是接口切片類型,這樣才能通過編譯,這也是官方推薦的做法
代碼如下所示:
// 獲得一個int類型的切片
var dataSlice []int = foo()
// 創(chuàng)建接口類型的切片
var interfaceSlice []interface{} = make([]interface{}, len(dataSlice))
// 依次轉換每個元素
for i, d := range dataSlice {
interfaceSlice[i] = d
}
// 調用上面的size方法
size(interfaceSlice)
如果按照上面的寫法來傳參,那么求切片的長度就顯得太費勁了,還不如直接調用 len(dataSlice) 完事。
事情發(fā)展到這里,有點不甘心,于是繼續(xù)查資料,發(fā)現(xiàn)go語言的反射機制可以解決這個問題。
首先將上面的size函數(shù)的參數(shù)改為接口類型(a interface{}),然后在里面用reflect.TypeOf(a).Kind()來判斷實參的類型,如果是切片類型,則用reflect.ValueOf()來獲得該切片,最后返回切片的長度
代碼如下所示:
func Size(a interface{}) int {
if reflect.TypeOf(a).Kind() != reflect.Slice {
return -1
}
ins := reflect.ValueOf(a)
return ins.Len()
}
補充:Go語言中切片的長度與容量的變化
在學習go語言的切片信息時,發(fā)現(xiàn)切片的容量變化非常讓人摸不著頭腦,為了更記憶深刻就寫下了這篇,如有錯誤之處,請大家指正
一、當前切片的長度與容量相等情況:
package main
import (
"fmt"
)
func main() {
numbers := []int{0,1,2}
printSlice(numbers)
//通過append給numbers增加信息,如果當前切片的長度與容量相等,增加信息的長度小于等于原來的長度,
那么切片的長度變?yōu)橄嗉又?,容量變?yōu)樵瓉淼?倍(圖片一的第二行結果)
numbers = append(numbers, 10,5,6)
printSlice(numbers)
//通過append給numbers增加信息,如果當前切片A的長度與容量相等,增加信息B的長度大于切片A原來的長度,
那么切片的長度變?yōu)橄嗉又停萘孔優(yōu)椋築長度+A長度+(B長度-A長度)%2(圖片一的第三行結果)
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)
}
圖片一:

二、如果當前切片的長度小于容量情況:
1、增加信息的長度與當前長度和小于等于容量
func main() {
/* 創(chuàng)建切片 */
numbers := []int{0,1,2}
printSlice(numbers)
numbers = append(numbers, 10,5)
printSlice(numbers)
//通過append給numbers增加信息,如果當前切片的長度小于容量,增加信息的長度與當前長度和小于等于容量,
那么numbers的長度變?yōu)橄嗉又?,容量不?圖片二的第三行結果)
numbers = append(numbers, 11)
printSlice(numbers)
}
圖片二:

2、增加信息B的長度與當前A的長度大于A容量并且小于A容量的2倍
func main() {
/* 創(chuàng)建切片 */
numbers := []int{0,1,2}
printSlice(numbers)
numbers = append(numbers, 10,5)
printSlice(numbers)
//通過append給numbers增加信息,如果當前切片A的長度小于容量,增加信息B的長度與當前A的長度大于A容量并且小于A容量的2倍,
那么numbers的長度變?yōu)橄嗉又?,容量變?yōu)椋篈容量*2(圖片三的第三行結果)
numbers = append(numbers, 11,12)
printSlice(numbers)
}
圖片三:

3、增加信息B的長度與當前A的長度大于A容量的2倍
func main() {
/* 創(chuàng)建切片 */
numbers := []int{0,1,2}
printSlice(numbers)
numbers = append(numbers, 10,5)
printSlice(numbers)
//通過append給numbers增加信息,如果當前切片A的長度小于容量,增加信息B的長度與當前A的長度大于A容量的2倍,
那么numbers的長度變?yōu)橄嗉又虲。容量變?yōu)椋築長度+A長度+(B長度-A長度)%2(圖片四的第三行結果)
numbers = append(numbers, 11,12,13,15,16,17,18,19,20)
printSlice(numbers)
}
圖片四:

以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。
您可能感興趣的文章:- 如何在Go中使用切片容量和長度
- Go語言切片前或中間插入項與內置copy()函數(shù)詳解
- golang中切片copy復制和等號復制的區(qū)別介紹
- go語言中切片與內存復制 memcpy 的實現(xiàn)操作
- go語言中的二維切片賦值
- go語言中切片的長度和容量的區(qū)別