重学Go语言之数组的具体使用详解

 更新时间:2023年02月28日 14:35:06   作者:程序员读书  
Go的数组是一种复合数据类型,在平时开发中并不常用,更常用的是切片(slice),可以把切片看作是能动态扩容的数组,切片的底层数据结构就是数组,所以数组虽不常用,但仍然有必要掌握

什么是数组

什么是数组?数组是有固定长度的相同数据类型元素的集合, 如下图所示:

从数组的定义以及上面的示例图我们可以得到数组的三个特征:

  • 固定长度,数组的长度在编译时就要确定。
  • 每个元素的数据类型相同。
  • 数组索引从0开始,索引的最大值为数组长度减1。

数组的创建

直接声明数组变量,在声明时必须指定长度:

var iArray [2]int
var sArray [3]string

上面的代码中,我们创建了一个长度为2,元素数据类型为int的数组和一个长度为3,元素数据类型为string的数组。

声明后,就可以通过索引给对应的元素赋值:

iArray[0] = 1
iArray[1] = 2
sArray[0] = "hello"
sArray[1] = "world"
sArray[2] = "!"

fmt.Println(iArray) //输出 [1,2]
fmt.Println(sArray) //输出 ["hello","world","!"]

也可以在声明时通过花括号{}直接初始化数组元素的值:

var iArray [2]int = [2]int{1,2}

//花括号内初始化元素数量可以少于数组的长度,后面没有初始化的元素会被赋予该数据类型的默认值
var sArray [4]string = [4]string{"A","B","C"} 

如果在声明时或者之后没有通过索引给数组的元素赋值,那么元素的值为对应数据类型的初始值:

var iArray [3]int 
var sArray [4]string
fmt.Println(iArray) //输出:[0,0,0]
fmt.Println(sArray) //输出:[]

通知短变量可以让数组声明的更简洁:

i := [2]int{1,2}

也可以在声明数组不指定数组长度,而是通过...和花括号{}内的初始化值让Go语言自动推断数组的长度:

var i = [...]int{1, 2, 3, 4} //数组长度为4

访问数组的元素

通过索引可以访问数组中的某个元素:

fmt.Println(iArray[0])

无论是给数组的元素赋值,还是访问数组的元素都不超过数组的长度,否则会数组越界的错误,数组的索引从0开始,因此数组的索引取值范围是0~len-1(len表示数组的长度)。

iArray := [2]int{1,2}
sArray := [3]string{"A","B","C"}
iArray[2] = 10 //报错,该数组索引的取值范围是0~1
fmt.Println(sArray[10]) // 报错,该数组索引的取值范围是0~2

数组的长度

Go内置的函数len()可以用于获得数组的长度:

iArray := [4]int{1,2,3,4}
fmt.Println(len(iArray)) // 输出结果:4

如何遍历数组

遍历数组使用for语句,有两种方式:

使用for语句遍历数组:

for i := 0; i < len(iArray); i++ {
	fmt.Println(iArray[i])
}

使用for-range遍历数组:

for k,v := range iArray {
	fmt.Println(k,v)
}

for-range遍历数组时,可以获取数组的索引和数组的元素,也可以在遍历时选择忽略索引或者元素值:

for _,v := range iArray { //忽略数组的索引
	fmt.Println(v)
}

for k,_ := range iArray{ //忽略元素
	fmt.Println(k)
}

数组的比较

数组只能进行相等(==)或者不相等(!=)的比较,并且两个进行比较的数组要符合以下要求,否则代码无法通过编译:

  • 数组元素的数据类型必须一致
  • 数组的长度必须一致

当数组满足上面的要求后,如果对应索引元素值相同,则数组相等,否则不相等:

iArray1 := [2]int{1, 2}
iArray2 := [2]int{1, 2}
if iArray1 == iArray2 {
    print("相等")
} else {
    print("不相等")
}
//输出:相等

iArray3 := [2]int{2, 1}
iArray4 := [2]int{1, 2}
if iArray1 == iArray2 {
    print("相等")
} else {
    print("不相等")
}

//输出:不相等

查找数组中的元素

对于数组来说,要查找数组中是否存在某个元素,并返回其对应索引,就要遍历一个数组,并对每个元素进行比较:

sArray := [5]string{"Java","PHP","Go","Python","JavaScript"}

for index, element := range sArray {
	if element == needle {
			fmt.Println(index)
		}
}

如果我们要查找的元素在数组的最后一个,那么要遍历整个数组才能查找到,查找元素的时间复杂度为O(n)

将数组作为函数参数

把数组作为参数传递给函数时,有几个注意的地方:

  • 当把数组作为参数传给函数时,Go会把数组复制一份传给函数,所以数组作为函数参数时是值传递而不是引用传递。
  • 数组作为参数,会被复制,因此如果传递的数组很大,复制就会很耗时。
  • 传递给函数的数组,其长度与数据类型必须函数形参一致,因此复用性很差。
func updateArray(haystack [5]int, index int, value int) error {
	if index >= len(haystack) {
		return errors.New("索引不能超过数组长度")
	}

	haystack[index] = value
	fmt.Println(haystack) //[1 100 3 4 5]
	return nil
}

func main() {
	iArray := [5]int{1, 2, 3, 4, 5}
	updateArray(iArray, 1, 100)
	fmt.Println(iArray) // [1 2 3 4 5]
}

上面这个例子中,我们希望updateArray函数可以修改我们指定索引的元素数组,但实际修改的复制后数组,与我们传给函数的数组无关,解决的办法是传递数组的指针:

func updateArray(haystack *[5]int, index int, value int) error {
	if index >= len(haystack) {
		return errors.New("索引不能超过数组长度")
	}

	haystack[index] = value
	fmt.Println(haystack) //[1 100 3 4 5]
	return nil
}

func main() {
	iArray := [5]int{1, 2, 3, 4, 5}
	updateArray(&iArray, 1, 100)
	fmt.Println(iArray) // [1 100 3 4 5]
}

虽然传递数组指针可以避免数组复制导致的性能问题,但是数组的长度和元素数据类型仍然要求一致,这大概就是数组不怎么被使用的原因吧:

func main() {
	iArray := [6]int{1, 2, 3, 4, 5} //把数组长度改为6
	updateArray(&iArray, 1, 100)  //报错
}

同理,当我们把一个数组变量赋值另外一个变量时,Go也是把数组复制一份给新的变量,如果想把新的变量指向原来的数组,同样是用数组的指针:

iArray := [2]int{1,2}

iArray1 := iArray

iArray[0] = 10

fmt.Println(iArray1) //输出:[10,2]
fmt.Println(iArray) //输出:[1,2]

iArray2 := &iArray
iArray2[0] = 20;

fmt.Println(iArray2) //输出:&[20,2]
fmt.Println(iArray) //输出:[20,2]

二维与多维数组

Go也支持二维和多维数组,其创建方式与一维数组类似:

二维数组:

iArrays := [3][2]string{{"A","B"},{"C","D"},{"E","F"}}

上述二维数组的结构如下图所示:

多维数组:

iArrays := [3][4][2]int{
		{
			{1, 2},
			{3, 4},
			{5, 6},
		},
		{
			{7, 8},
			{9, 10},
			{11, 12},
		},
		{
			{13, 14},
			{15, 16},
			{17, 18},
		},
	}

上述三维数组的结构如下图所示:

小结

总结一下,这篇文章主要讲了以下几点:

  • 数组是一种固定长度的相同数据类型元素的集合,在实际开发中并不常用,而是作为slice的底层数据结构。
  • Go支持一维、二维和多维数组
  • 数组可以进行相等或者不相等的比较
  • 使用for或者for-range可以遍历数组
  • 通过数组索引访问元素或者给元素赋值时,都不能超过数组的长度限制。

到此这篇关于重学Go语言之数组的具体使用详解的文章就介绍到这了,更多相关Go语言 数组内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 一文带你搞懂Golang如何正确退出Goroutine

    一文带你搞懂Golang如何正确退出Goroutine

    在Go语言中,Goroutine是一种轻量级线程,它的退出机制对于并发编程至关重要,下午就来介绍几种Goroutine的退出机制,希望对大家有所帮助
    2023-06-06
  • 一文带你掌握Golang基础之通道

    一文带你掌握Golang基础之通道

    在Java中,多线程之间的通信方式有哪些?记得吗?Java多线程间通信的解决方案有很多种,比如:synchronized。在go中,就一种:通道,文中介绍的非常详细,感兴趣的同学可以参考下
    2023-05-05
  • golang快速实现网页截图的方法

    golang快速实现网页截图的方法

    这篇文章主要介绍了golang快速实现网页截图的方法,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-03-03
  • Go实现将任何网页转化为PDF

    Go实现将任何网页转化为PDF

    在许多应用场景中,可能需要将网页内容转化为 PDF 格式,使用Go编程语言,结合一些现有的库,可以非常方便地实现这一功能,下面我们就来看看具体实现方法吧
    2024-11-11
  • Go1.18 新特性之多模块Multi-Module工作区模式

    Go1.18 新特性之多模块Multi-Module工作区模式

    这篇文章主要介绍了Go1.18 新特性之多模块Multi-Module工作区模式,在 Go 1.18之前,建议使用依赖模块中的 replace 指令来处理这个问题,从 Go 1.18开始引入了一种同时处理多个模块的新方法,通过案例给大家详细介绍,感兴趣的朋友一起看看吧
    2022-04-04
  • Go中的错误和异常处理最佳实践方法

    Go中的错误和异常处理最佳实践方法

    这篇文章主要介绍了Go中的错误和异常处理最佳实践方法,文章围绕主题展开详细的内容戒杀,具有一定的参考价值,需要的小伙伴可以参考一下
    2022-09-09
  • Go语言实现字符串切片赋值的方法小结

    Go语言实现字符串切片赋值的方法小结

    这篇文章主要给大家介绍了Go语言实现字符串切片赋值的两种方法,分别是在for循环的range中以及在函数的参数传递中实现,有需要的朋友们可以根据自己的需要选择使用。下面来一起看看吧。
    2016-10-10
  • Go编程中常见错误和不良实践解析

    Go编程中常见错误和不良实践解析

    这篇文章主要为大家介绍了Go编程中常见错误和不良实践解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2024-01-01
  • Golang 使用接口实现泛型的方法示例

    Golang 使用接口实现泛型的方法示例

    这篇文章主要介绍了Golang 使用接口实现泛型的方法示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-03-03
  • 一文带你学会Go select语句轻松实现高效并发

    一文带你学会Go select语句轻松实现高效并发

    这篇文章主要为大家详细介绍了Golang中select语句的用法,文中的示例代码讲解详细,对我们学习Golang有一定的帮助,需要的可以参考一下
    2023-03-03

最新评论