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 相关文章推荐
Go语言-为什么返回值为接口类型,却返回结构体
Apr 24 Golang
Go语言使用select{}阻塞main函数介绍
Apr 25 Golang
golang 在windows中设置环境变量的操作
Apr 29 Golang
解决Golang time.Parse和time.Format的时区问题
Apr 29 Golang
Golang 如何实现函数的任意类型传参
Apr 29 Golang
go设置多个GOPATH的方式
May 05 Golang
go goroutine 怎样进行错误处理
Jul 16 Golang
Golang 并发下的问题定位及解决方案
Mar 16 Golang
Golang Elasticsearches 批量修改查询及发送MQ
Apr 19 Golang
Golang 对es的操作实例
Apr 20 Golang
Golang jwt身份认证
Apr 20 Golang
Go调用Rust方法及外部函数接口前置
Jun 14 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
150kHz到30Mhz完全冲浪手册
2020/03/20 无线电
多文件上传的例子
2006/10/09 PHP
windows下PHP_intl.dll正确配置方法(apache2.2+php5.3.5)
2014/01/14 PHP
Yii2前后台分离及migrate使用(七)
2016/05/04 PHP
详谈PHP程序Laravel 5框架的优化技巧
2016/07/18 PHP
Laravel的throttle中间件失效问题解决方法
2016/10/09 PHP
laravel 字段格式化 modle 字段类型转换方法
2019/09/30 PHP
jQuery 全选效果实现代码
2009/03/23 Javascript
兼容ie、firefox的图片自动缩放的css跟js代码分享
2013/08/12 Javascript
告诉你什么是javascript的回调函数
2014/09/04 Javascript
学习JavaScript设计模式(接口)
2015/11/26 Javascript
js实现html table 行,列锁定的简单实例
2016/10/13 Javascript
详解Angular.js的$q.defer()服务异步处理
2016/11/06 Javascript
jQuery实现文档树效果
2017/02/20 Javascript
ajax与jsonp的区别及用法
2018/10/16 Javascript
react-router 路由切换动画的实现示例
2018/12/03 Javascript
JS中的防抖与节流及作用详解
2019/04/01 Javascript
详解jquery和vue对比
2019/04/16 jQuery
vue 使用高德地图vue-amap组件过程解析
2019/09/07 Javascript
解决Vue-cli无法编译es6的问题
2020/10/30 Javascript
python分割列表(list)的方法示例
2017/05/07 Python
获取Django项目的全部url方法详解
2017/10/26 Python
python opencv设置摄像头分辨率以及各个参数的方法
2018/04/02 Python
django输出html内容的实例
2018/05/27 Python
python爬虫的一个常见简单js反爬详解
2019/07/09 Python
Python如何向SQLServer存储二进制图片
2020/06/08 Python
经典c++面试题五
2014/12/17 面试题
汽车专业学生自我评价
2014/01/19 职场文书
党员公开承诺书
2014/03/25 职场文书
活动总结格式范文
2014/04/26 职场文书
青春雷锋观后感
2015/06/10 职场文书
幼儿园教师培训心得体会
2016/01/21 职场文书
一起来看看Vue的核心原理剖析
2022/03/24 Vue.js
Redis如何实现验证码发送 以及限制每日发送次数
2022/04/18 Redis
Nginx安装配置详解
2022/06/25 Servers
Go语言编译原理之源码调试
2022/08/05 Golang