脚本之家,脚本语言编程技术及教程分享平台!
分类导航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服务器之家 - 脚本之家 - Golang - go语言中切片Slice与数组Array对比以及panic: runtime error: index out of range问题解决

go语言中切片Slice与数组Array对比以及panic: runtime error: index out of range问题解决

2022-07-20 13:32大锤爱编程 Golang

go语言中数组与其他语言有在显著的不同,包括其不能够进行添加,以及值拷贝的特性,下面这篇文章主要给大家介绍了关于go语言中切片Slice与数组Array对比以及panic: runtime error: index out of range问题解决的相关资料,需要的朋友可以参考下

前言

在go语言的学习历程当中,slice数据类型引起了我的好奇心。为啥不直接使用Slice,是人性的扭曲还是道德的沦丧~,下面让我们一探究竟~~

一、go slice是什么

go语言中的slice是一个基于Array封装的数据结构,go语言中slice的使用频率远高于array,其身影频频出现在源码实现当中。slice相对于Array的优点就是其可以动态调整自己的size,不像Array的Size是固定的。

二、go slice实战案例

1.slice创建、使用

slice的创建有两个方法分别是使用字面量定义和使用make函数。除过slice创建,其他slice的生成办法均是从现有slice分片或者array上做slice分片操作。

slice创建代码:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
package main
 
import (
    "fmt"
    "reflect"
)
 
func main() {
    //字面创造Slice
    sliceOne := []string{"a", "b"}
    //使用make函数创造slice
    sliceTwo := make([]string, 10)
    sliceThree := make([]int, 10)
 
    fmt.Printf("使用字面量创建的slice%s\n",reflect.ValueOf(sliceOne).String())
    fmt.Printf("使用make函数创建的slice:%s\n",reflect.ValueOf(sliceTwo).String())
    fmt.Printf("使用make函数创建的slice:%s\n",reflect.ValueOf(sliceThree).String())
 
}

程序输出:

使用字面量创建的slice<[]string Value>
使用make函数创建的slice:<[]string Value>
使用make函数创建的slice:<[]int Value>
 
Process finished with the exit code 0

2、slice的长度和容量概念理解

学习过程中,很多小伙伴会对slice的长度和容量问题有着很多混淆。

这个地方可以把切片比喻成一个可以装10个苹果的袋子,现在的袋子里面有三颗苹果。切片的长度就是袋子已经装的果子的个数,目前是3个。切片的容量就是这个袋子一共能装多少个果子,对于这个袋子来说就是10。那么把代码替换成切片,把苹果替换成元素,是不是就懂了撒~

下面就是该问题的处理办法就是直接去官方,看源码。看看第一手资料怎么讲!

长度:slice中拥有的元素个数,如果slice是nil的话,则元素个数长度是0
英文:the number of elements in v; if v is nil, len(v) is zero

容量:slice切片的长度能够到达的最大值
英文:Slice: the maximum length the slice can reach when resliced;

代码验证环节:

?
1
2
3
4
5
6
7
8
9
10
11
12
package main
 
import (
    "fmt"
 )
 
func main() {
    sliceOne := []string{"a", "b"}
    strings := sliceOne[0:1]
    fmt.Printf("切片的长度:%d\n",len(strings))
    fmt.Printf("切片的容量:%d\n",cap(strings))
}

代码结果输出:

切片的长度:1
切片的容量:2

代码原理解析:

strings由sliceOne切片而来,切出来的片上数据有的是0到1,有一个元素,故其对应的长度是1。

因为切片是一个引用类型,只在原始切片上切出了0到1的位置,剩余的空位还有1,故其容量等于长度加剩余元素位置数。

3. 切片扩容及slice panic: runtime error: index out of range

slice越界代码实例如下:

?
1
2
3
4
sliceOne := []string{"a", "b"}
//使用make函数创造slice
s := sliceOne[2]
fmt.Printf(s)

使用sliceOne[2]语句时,数组越界报错。

实际开发过程中,总会有slice容量不够用的时候,该怎么扩容,如何保证安全扩容?

go语言官方提供的扩容办法就是创建一个新的更大的分片,将老分片的数据内容迁移到新的切片当中。

代码展示:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
package main
 
import (
    "fmt"
 )
 
func main() {
    sliceOne := []string{"a", "b"}
    fmt.Printf("切片扩容前")
    fmt.Printf("切片的长度:%d\n",len(sliceOne))
    fmt.Printf("切片的容量:%d\n",cap(sliceOne))
    t := make([]string, len(sliceOne), (cap(sliceOne))*2)
    copy(t, sliceOne)
    sliceOne = t
    fmt.Printf("切片扩容后")
    fmt.Printf("切片的长度:%d\n",len(sliceOne))
    fmt.Printf("切片的容量:%d\n",cap(sliceOne))
}

结果展示:

切片的长度:2
切片的容量:2
切片的长度:2
切片的容量:4

从代码结果上看出新切片的长度是2,容量是4,也再次验证了切片的长度取决于存放了多少元素,切片的容量取决于已存放的元素数量加剩余位置数。

附:go 判断数组下标是否存在

举例

现在需要判断命令行是否传了参数,即 os.Args[1] 是否存在

如果使用下述的判断:

?
1
2
3
func main() {
    fmt.Println(os.Args[1])
}

会报错:index out of range

panic: runtime error: index out of range [1] with length 1
 
goroutine 1 [running]:
main.main()
        D:/go_work/test/test4.go:9 +0xbc
exit status 2

现有两种方式解决:

第一种:

通过遍历的方式判断 key 是否存在

?
1
2
3
4
5
6
7
8
9
10
11
func main() {
    var result string
    for k, v := range os.Args {
        if k == 1 {
            result = v
        }
    }
    if result != "" {
        fmt.Printf("os.Args[1] = %s", result)
    }
}

第二种:

由于数组下标从0开始,len(arr)-1 为最后一个元素的下标,所以判断所要查询的 key 是否小于 len(arr) 就可以了

?
1
2
3
4
5
func main() {
    if len(os.Args)>=2 {
        fmt.Printf("os.Args[1] = %s", os.Args[1])
    }
}

总结

go语言中slice的应用和使用相对来说方便快捷很多,不过也有一些小小的暗坑等待大家发现和整理哦~后续我会在我的博客中,继续发布有关于go语言使用的tips和技巧~

到此这篇关于go语言中切片Slice与数组Array对比以及panic: runtime error: index out of range问题解决的文章就介绍到这了,更多相关go语言切片Slice和数组Array内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://blog.csdn.net/alike_u/article/details/125436734

延伸 · 阅读

精彩推荐
  • Golanggolang 并发安全Map以及分段锁的实现方法

    golang 并发安全Map以及分段锁的实现方法

    这篇文章主要介绍了golang 并发安全Map以及分段锁的实现方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧 ...

    薛薛薛3702020-05-23
  • Golanggolang如何使用sarama访问kafka

    golang如何使用sarama访问kafka

    这篇文章主要介绍了golang如何使用sarama访问kafka,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧 ...

    CodingCode5192020-05-22
  • Golang你对 Go 错误处理的 4 个误解!

    你对 Go 错误处理的 4 个误解!

    Go 语言中错误处理的机制一直是各大 Gopher 热议的问题。甚至一直有人寄望 Go 支持 throw 和 catch 关键字,实现与其他语言类似的特性。社区里的讨论也从未...

    脑子进煎鱼了7512021-09-14
  • Golanggo语言实战之实现比特币地址校验步骤

    go语言实战之实现比特币地址校验步骤

    这篇文章主要介绍了go语言实战之实现比特币地址校验步骤,利用生产的随机数采用椭圆加密算法生成公钥,具体步骤实例代码请参考下本文...

    m0_377190479082021-06-26
  • Golang基于golang如何实现error工具包详解

    基于golang如何实现error工具包详解

    Go 语言使用 error 类型来返回函数执行过程中遇到的错误,下面这篇文章主要给大家介绍了关于如何基于golang实现error工具包的相关资料,文中通过示例代码...

    zcv85702020-05-19
  • Golanggolang实现多协程下载文件(支持断点续传)

    golang实现多协程下载文件(支持断点续传)

    本文主要介绍了golang实现多协程下载文件,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    山与路10162021-12-02
  • Golanggolang协程池设计详解

    golang协程池设计详解

    这篇文章主要介绍了golang协程池设计详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小...

    薛薛薛4432020-05-28
  • Golanggolang jwt+token验证的实现

    golang jwt+token验证的实现

    这篇文章主要介绍了golang jwt+token验证的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着...

    诺尔尼亚之星17062021-01-30