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

 更新时间:2022年07月20日 10:28:31   作者:大锤爱编程  
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创建代码:

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;

代码验证环节:

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越界代码实例如下:

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

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

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

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

代码展示:

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] 是否存在

如果使用下述的判断:

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 是否存在

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) 就可以了

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内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Golang gRPC HTTP协议转换示例

    Golang gRPC HTTP协议转换示例

    这篇文章主要为大家介绍了Golang gRPC HTTP协议转换示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-06-06
  • Go标准库http与fasthttp服务端性能对比场景分析

    Go标准库http与fasthttp服务端性能对比场景分析

    这篇文章主要介绍了Go标准库http与fasthttp服务端性能比较,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-06-06
  • Go 如何基于IP限制HTTP访问频率的方法实现

    Go 如何基于IP限制HTTP访问频率的方法实现

    这篇文章主要介绍了Go 如何基于IP限制HTTP访问频率的方法实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-11-11
  • GoLang中拼接字符串性能优化方法详解

    GoLang中拼接字符串性能优化方法详解

    最近在做性能优化,有个函数里面的耗时特别长,看里面的操作大多是一些字符串拼接的操作,而字符串拼接在 golang 里面其实有很多种实现,下面这篇文章主要给大家介绍了关于Golang语言如何高效拼接字符串的相关资料,需要的朋友可以参考下
    2023-02-02
  • golang解析yaml文件操作

    golang解析yaml文件操作

    这篇文章主要介绍了golang解析yaml文件操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-12-12
  • 详解如何在Go中循环中使用Defer关键字示例详解

    详解如何在Go中循环中使用Defer关键字示例详解

    这篇文章主要为大家介绍了详解如何在Go中循环中使用Defer关键字示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-09-09
  • 详解Golang中interface接口的原理和使用技巧

    详解Golang中interface接口的原理和使用技巧

    interface 接口在 Go 语言里面的地位非常重要,是一个非常重要的数据结构。本文主要介绍了Golang中interface接口的原理和使用技巧,希望对大家有所帮助
    2022-11-11
  • golang监听ip数据包的实现步骤(golang纯享版)

    golang监听ip数据包的实现步骤(golang纯享版)

    这篇文章主要给大家介绍了golang监听ip数据包的实现步骤,本文以ip4 作为案例进行包抓取示范,ip6抓取与ip4方式异曲同工,可自行举一反三得出,文中通过图文结合给大家介绍的非常详细,需要的朋友可以参考下
    2024-02-02
  • golang beego框架路由ORM增删改查完整案例

    golang beego框架路由ORM增删改查完整案例

    这篇文章主要为大家介绍了golang beego框架路由ORM增删改查完整案例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步早日升职加薪
    2022-04-04
  • go开源项目用户名密码验证的逻辑鬼才写法

    go开源项目用户名密码验证的逻辑鬼才写法

    这篇文章主要为大家介绍了go开源项目中发现的一个逻辑鬼才写法,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-07-07

最新评论