go语言求任意类型切片的长度操作


Posted in Golang onApril 26, 2021

最近用go写程序时遇到一个问题——求任意类型切片的长度。

作为一个初学者,刚刚学了接口和切片,知道了每个类型都实现了一个空接口interface{},那么如果接口类型作为函数的参数,那它应该是可以接收任意类型的实参的

带着这样的想法就写出了下面的代码:

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()
// 创建接口类型的切片
var interfaceSlice []interface{} = make([]interface{}, len(dataSlice))
// 依次转换每个元素
for i, d := range dataSlice {
 interfaceSlice[i] = d
}
// 调用上面的size方法
size(interfaceSlice)

如果按照上面的写法来传参,那么求切片的长度就显得太费劲了,还不如直接调用 len(dataSlice) 完事。

事情发展到这里,有点不甘心,于是继续查资料,发现go语言的反射机制可以解决这个问题。

首先将上面的size函数的参数改为接口类型(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语言的切片信息时,发现切片的容量变化非常让人摸不着头脑,为了更记忆深刻就写下了这篇,如有错误之处,请大家指正

一、当前切片的长度与容量相等情况:

package main
import (
    "fmt"
)
func main() {
	numbers := []int{0,1,2}  
	printSlice(numbers)
	//通过append给numbers增加信息,如果当前切片的长度与容量相等,增加信息的长度小于等于原来的长度,
	那么切片的长度变为相加之和,容量变为原来的2倍(图片一的第二行结果)
	numbers = append(numbers, 10,5,6) 
	printSlice(numbers)
	
	//通过append给numbers增加信息,如果当前切片A的长度与容量相等,增加信息B的长度大于切片A原来的长度,
	那么切片的长度变为相加之和,容量变为:B长度+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)
}

图片一:

go语言求任意类型切片的长度操作

二、如果当前切片的长度小于容量情况:

1、增加信息的长度与当前长度和小于等于容量

func main() {
   /* 创建切片 */
   numbers := []int{0,1,2}  
   printSlice(numbers)
   numbers = append(numbers, 10,5) 
   printSlice(numbers)
   //通过append给numbers增加信息,如果当前切片的长度小于容量,增加信息的长度与当前长度和小于等于容量,
   那么numbers的长度变为相加之和,容量不变(图片二的第三行结果)
   numbers = append(numbers, 11)  
   printSlice(numbers)
}

图片二:

go语言求任意类型切片的长度操作

2、增加信息B的长度与当前A的长度大于A容量并且小于A容量的2倍

func main() {
   /* 创建切片 */
   numbers := []int{0,1,2}  
   printSlice(numbers)
   numbers = append(numbers, 10,5) 
   printSlice(numbers)
   //通过append给numbers增加信息,如果当前切片A的长度小于容量,增加信息B的长度与当前A的长度大于A容量并且小于A容量的2倍,
   那么numbers的长度变为相加之和,容量变为:A容量*2(图片三的第三行结果)
   numbers = append(numbers, 11,12)  
   printSlice(numbers)
}

图片三:

go语言求任意类型切片的长度操作

3、增加信息B的长度与当前A的长度大于A容量的2倍

func main() {
	/* 创建切片 */
	numbers := []int{0,1,2}  
	printSlice(numbers)
	numbers = append(numbers, 10,5) 
	printSlice(numbers)
	//通过append给numbers增加信息,如果当前切片A的长度小于容量,增加信息B的长度与当前A的长度大于A容量的2倍,
	那么numbers的长度变为相加之和C。容量变为:B长度+A长度+(B长度-A长度)%2(图片四的第三行结果)
	numbers = append(numbers, 11,12,13,15,16,17,18,19,20)  
	printSlice(numbers)
}

图片四:

go语言求任意类型切片的长度操作

以上为个人经验,希望能给大家一个参考,也希望大家多多支持三水点靠木。如有错误或未考虑完全的地方,望不吝赐教。

Golang 相关文章推荐
Golang 正则匹配效率详解
Apr 25 Golang
golang DNS服务器的简单实现操作
Apr 30 Golang
解决Goland 同一个package中函数互相调用的问题
May 06 Golang
go mod 安装依赖 unkown revision问题的解决方案
May 06 Golang
聊聊golang中多个defer的执行顺序
May 08 Golang
Golang二维数组的使用方式
May 28 Golang
修改并编译golang源码的操作步骤
Jul 25 Golang
浅谈GO中的Channel以及死锁的造成
Mar 18 Golang
Go归并排序算法的实现方法
Apr 06 Golang
Golang原生rpc(rpc服务端源码解读)
Apr 07 Golang
Golang 结构体数据集合
Apr 22 Golang
基于Python实现西西成语接龙小助手
Aug 05 Golang
golang如何去除多余空白字符(含制表符)
Apr 25 #Golang
用golang如何替换某个文件中的字符串
Apr 25 #Golang
Golang 正则匹配效率详解
golang正则之命名分组方式
Apr 25 #Golang
go语言-在mac下brew升级golang
Apr 25 #Golang
go原生库的中bytes.Buffer用法
Apr 25 #Golang
Go缓冲channel和非缓冲channel的区别说明
Apr 25 #Golang
You might like
PHP对象实例化单例方法
2017/01/19 PHP
THINKPHP截取中文字符串函数实例代码
2017/03/20 PHP
PHP编程中的Session阻塞问题与解决方法分析
2017/08/07 PHP
浅谈PHP SHA1withRSA加密生成签名及验签
2019/03/18 PHP
javascript 多级checkbox选择效果
2009/08/20 Javascript
Google的跟踪代码 动态加载js代码方法应用
2012/11/12 Javascript
jquery 获取表单元素里面的值示例代码
2013/07/28 Javascript
document.documentElement和document.body区别介绍
2013/09/16 Javascript
jquery自动切换tabs选项卡的具体实现
2013/12/24 Javascript
js常用数组操作方法简明总结
2014/06/20 Javascript
JavaScript中使用arguments获得函数传参个数实例
2014/08/27 Javascript
2014年50个程序员最适用的免费JQuery插件
2014/12/15 Javascript
浅谈javascript中for in 和 for each in的区别
2015/04/23 Javascript
原生JS和JQuery动态添加、删除表格行的方法
2015/05/28 Javascript
jquery中的工具使用方法$.isFunction, $.isArray(), $.isWindow()
2015/08/09 Javascript
原生js配合cookie制作保存路径的拖拽
2015/12/29 Javascript
Bootstrap树形组件jqTree的简单封装
2016/01/25 Javascript
JS简单判断函数是否存在的方法
2017/02/13 Javascript
Nodejs 获取时间加手机标识的32位标识实现代码
2017/03/07 NodeJs
一篇看懂vuejs的状态管理神器 vuex状态管理模式
2017/04/20 Javascript
React教程之Props验证的具体用法(Props Validation)
2017/09/04 Javascript
angular.js4使用 RxJS 处理多个 Http 请求
2017/09/23 Javascript
JS关于刷新页面的相关总结
2018/05/09 Javascript
jQuery仿移动端支付宝键盘的实现代码
2018/08/15 jQuery
微信小程序本地存储实现每日签到、连续签到功能
2019/10/09 Javascript
在Python的Flask中使用WTForms表单框架的基础教程
2016/06/07 Python
Python实现邮件的批量发送的示例代码
2018/01/23 Python
python并发编程多进程 互斥锁原理解析
2019/08/20 Python
python tkinter图形界面代码统计工具
2019/09/18 Python
python itsdangerous模块的具体使用方法
2020/02/17 Python
高职助产应届生自荐信
2013/09/24 职场文书
2014年入党积极分子党课学习心得体会模板
2014/04/03 职场文书
2014年党支部学习材料
2014/05/19 职场文书
销售员岗位职责
2014/06/09 职场文书
现场施工员岗位职责
2015/04/11 职场文书
运动会开幕式致辞
2015/07/29 职场文书