Golang入门笔记(15)—— 数组和切片

        编程的世界中,或许是因为一次一次的定义变量,维护管理起来都太费劲了,所以推出了数组,将数据用数组的形式管理起来。

        Go的数组和Java的实现机制是不同的,Go语言的数组是作为基本数据类型存在的。所以数组是开辟在栈帧中开辟内存的;在函数参数传递的时候是,也是当作基本数据类型,进行了值传递。

  •        数组

        先不讲那么多七七八八的理论,来段代码直观感受一下Go语言的数组:

package main

import "fmt"

func main() {

	var scores [5]int
	scores[0] = 90
	scores[1] = 91
	scores[2] = 92
	scores[3] = 93
	scores[4] = 94

	//求和,求平均
	sum := 0
	for i := 0; i < len(scores); i++ {
		sum += scores[i]
	}
	fmt.Println(sum)
	fmt.Println(sum / len(scores))

	fmt.Printf("scores 的地址,即第一个空间对应的地址 , %p \n", &scores)
	println("-------------- 16进制,每次大小间隔为8 ------------------")
	fmt.Printf("scores 第0个空间对应的地址 , %p \n", &scores[0])//scores 第0个空间对应的地址 , 0xc000010450
	fmt.Printf("scores 第1个空间对应的地址 , %p \n", &scores[1])//scores 第1个空间对应的地址 , 0xc000010458
	fmt.Printf("scores 第2个空间对应的地址 , %p \n", &scores[2])//scores 第2个空间对应的地址 , 0xc000010460
	fmt.Printf("scores 第3个空间对应的地址 , %p \n", &scores[3])//scores 第3个空间对应的地址 , 0xc000010468
	fmt.Printf("scores 第4个空间对应的地址 , %p \n", &scores[4])//scores 第4个空间对应的地址 , 0xc000010470
}

具体分析:

        16进制数,数组位置,每次大小偏移都是间隔为8,这个8就是每个元素的空间大小。         

关于更多关于数组的认知,可参考之前写的一篇帖子:CSDN

        下面做一个小demo:

package main

import "fmt"

func main() {

	//已知,班级中有10位同学,录入各同学成绩,计算得出总分和平均分。
	var scores [10]int
	//求和,求平均
	sum := 0
	for i := 0; i < len(scores); i++ {
		fmt.Printf("\n 请录入第%v个学生的成绩 \n", i+1)
		fmt.Scanln(&scores[i]) // 录入,改变数组中元素i的值,通过 & 绑定到哪个地址。
		sum += scores[i]
	}
	fmt.Println("总分:", sum)
	fmt.Println("平均分", sum/len(scores))

}

        数据遍历方式,除了使用上面代码中的以索引下标进行普通遍历外,还可以使用for range 遍历,for range 可以遍历数组,切片,字符串,map 以及通道,for range语法上类似于其他编程语法的foreach语句,形式如下:

for key,val:= range collection{
    ...
}

           代码如下:

package main

func main() {

	var scores [5]int
	scores[0] = 90
	scores[1] = 91
	scores[2] = 92

	for key, value := range scores {
        //key,value 属于当次循环的局部变量。
		println("[", key, "] ==> ", value)
	}


	for key, value := range scores {
        //key,value 属于当次循环的局部变量。
		println("[", key, "] ==> ", value)
	}

    // 如果用不到key,使用 下划线 “_” 可以忽略。
	for _, value := range scores {
		println( "==> ", value)
	}
}

     key,value 属于当次循环中的局部变量。

数组的初始化方式:

第一种:var 类型自己推断。

	var scores = [5] int {1,3,5,7,9}

第二种:事先不知道开辟多大空间。

	var scores = [...]int{1, 3, 5, 7, 9}

第三种:根据下标逐一赋值

	var scores = [...]int{0: 70, 2: 30, 3: 99, 4: 100}

数组的注意事项:

1.数组的长度属于类型的一部分,相当于动态数组的类属性。

	var a = [...]int{0: 70, 2: 30, 3: 99, 4: 100}
	fmt.Printf(" 数组类型:%T", a) // [5]int

2. Java中数组属于引用类型,但是在Go语言中,数组属于值类型的,默认情况下,按值传递,因此会进行值拷贝。如果想在其他函数中,去修改原来的数组,可以通过指针的方式进行引用传递。


import "fmt"

func main() {

	var a = [...]int{0: 0, 1: 1, 2: 2, 3: 3, 4: 4}
	modify(a)
	fmt.Println(a)

	modifyPlus(&a) //传入地址
	fmt.Println(a)

}

func modify(arr [5]int) {  
	arr[0] = -999
}

func modifyPlus(arr *[5]int) { // star在类型Type上,means This‘s pointer the address (ptr)
	(*arr)[0] = -999 // 
}

        分析:当goroutine(轻量的go线程) 的 active-frame活动栈帧  从main函数 运行 到modifyPlus 函数的 栈帧时,通过 *pointerTheAddress变量ptr的方式,进行引用传递的方式,完成了对该内存数据的重新赋值;而modify 是默认的值传递,是值拷贝,操作不是同一块内存。

二维数组 Two Dimensional Array (注音:dimensional大爱闷声捞 恶ray )

        定义二维数组  var twoDimensionalArr  [2][3] int16  

        2个3长度的数组,默认初始值 [0,0,0],[0,0,0] ,内存分配,如下图:

var arr [2][3] int16 内存分配 示意图

        Anyhow无论 它是二维也好,N维也罢,在内存中都是一个连续的内存空间,就是在连续的数组中不断的拆分内存。 二维 [2][3] 即 开辟2*3 =6个类型大小。三维[2][3][4],即2*3*4=24 个类型大小。

package main

import (
	"fmt"
)

func main() {

	var a [2][3]int
	a[0][1] = 22
	fmt.Printf("\n\t a ↓ \n  %v \n", a) //  [ [0 22 0]  [0 0 0] ]

	var init2dArray [2][3]int = [2][3]int{{1, 2, 3}, {2, 3, 4}}
	fmt.Printf("二维数组的初始化操作 ==>  %v \n", init2dArray)

	fmt.Println("--------- 遍历方式1 普通for循环 ----------")
	for i := 0; i < len(init2dArray); i++ {
		for j := 0; j < len(init2dArray[i]); j++ {
			fmt.Println(init2dArray[i][j])
		}
	}

	fmt.Println("--------- 遍历方式2 for range循环 ----------")
	for outerKey, outerValue := range init2dArray {
		//fmt.Println(outerKey, outerValue)
		for innerKey, innerValue := range outerValue {
			fmt.Printf("arr[%v][%v]=[%v] \t", outerKey, innerKey, innerValue)
		}
		println()
	}

}

        同样,Go和Java一样,都支持多维数组:

package main

import (
	"fmt"
)

func main() {
	var a [2][3][4][5]int
	fmt.Printf("a ==>\n  %v ", a)
}
  •         切片

        切片可以算是对数组的扩展,数组的长度是固定的,所以在Go语言的代码的出现频率不高,而切片是一种建立在数组的基础上做了一层封装,提供了更加强大的功能。正因为如此,所以切片是一种引用类型。切片slice 是对数组一个连续片段的引用,所以切片是一个引用类型,对切片索引的操作,可以改变原来的数组索引上的元素;具体形式:[start:end],这个片段是由起始切点偏移索引和终止切点偏移索引构成,可以是整个数组,或者是部分数组,但是注意一点:切点偏移索引不能越界,切片感觉类似于Java的封装的ArrayList动态数组。

        切片三个要素之我的理解:切片的地址(首元素的地址?),实际元素大小size 和 capacity总的容量;

内存中 原数组 intArr 和切片 示意图

         

        切片定义的三种方式:

方式1:定义一个切片,然后让切片去 切(引用)一个以及创建好的数组。

	var strings []string = []string{"a", "b", "c", "d", "e", "f", "g", "h"}
	var stringSlice []string
	stringSlice = strings[2:6]

方式2:通过内置的make函数来创建切片。

基本语法:var 切片变量名 [] T类型 = make([]T类型,size,capacity);

         make([]T, size, cap)  底层数组的指针、切片的大小( len )和切片的容量(cap)。

	fmt.Println("-----------  make([]T, size, cap) --------------")
	var bridgeSlice []int = make([]int, 2, 10) 
	fmt.Println(len(bridgeSlice))              // len表示切片存储元素的长度
	fmt.Println(cap(bridgeSlice))              // cap表示最大可以存储的容量
	//ps 底层数组对外不可见,不能直接操作数组,通过操作slice切片变量桥接到数组。
	bridgeSlice[0] = 0
	bridgeSlice[1] = 1

        点评:底层数组对外不可见,不能直接操作数组,通过操作slice切片变量桥接到数组。

方式3:定义一个切片,直接就指定具体的数组,使用原理类似make方式。

	sliceDefine3 := []string{"a", "b", "c"}
	fmt.Println(len(sliceDefine3)) // len表示切片存储元素的长度
	fmt.Println(cap(sliceDefine3)) // cap表示最大可以存储的容量

        点评:质疑一下,我感觉和数组没啥区别呀,连实际元素len和capacity 都一样。不过,学习阶段,被动接受吧,别人说是啥,那就是啥。就算错了,以后再改了这段文字就好了。不追求完美,追求完美本身就是不完美。

参考代码:

	fmt.Println("----------- make函数  make([]T, size, cap) --------------")
	var bridgeSlice []int = make([]int, 2, 10) //底层数组的指针、切片的大小( len )和切片的容量(cap)。
	fmt.Println(len(bridgeSlice))              // len表示切片存储元素的长度
	fmt.Println(cap(bridgeSlice))              // cap表示最大可以存储的容量
	//ps 对外不可见,不能直接操作数组,通过操作slice切片变量桥接到数组。
	bridgeSlice[0] = 0
	bridgeSlice[1] = 1

切片的遍历:和数组遍历方式一样,普通for循环和for...range循环都可以。代码略,参考上面数组遍历。

切片的注意事项:

1,切片定义后,不能直接使用,需要让其引用到一个数组 或 make一个空间供切片使用。

2.切片不能越界。

3.切片可以继续切片。

4.切片的capacity 可以动态增长,容量不够,底层会创建一个新数组,然后指向新数组。该新数组不能直接维护,通过切片间接维护。如果你要想给原来切片append函数追加给原来的slice,并赋值给slice。

slice = append(slice,1,2,3)

5.切片的拷贝。copy函数,第一个参数: destination目标,第二个参数:source 源。

fmt.Println("----------- copy函数 --------------")
	var x1 []int = []int{1, 2, 3, 4, 5}
	var x2 []int = make([]int, 10)
	copy(x2, x1)
	fmt.Println(x2)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Golang数组切片是两种不同的数据类型,用于存储相同数据类型的容器。数组的长度是固定的,而切片的长度是可变的。在日常应用中,切片的使用更为普遍。 数组在声明时需要指定长度,并且在初始化时必须提供相同长度的元素。例如,`a := int{1, 2, 3}`就是一个长度为3的整数数组数组的长度一旦确定后就不能更改。 切片是基于数组的引用类型。它不需要指定固定的长度,并且可以根据需要动态扩展或缩小。切片包装着底层数组,通过指定起始索引和结束索引来指定子集。例如,`b := a[:]`就是一个切片,它包含了数组a的所有元素。 数组适用于需要固定长度的场景,而切片适用于长度可变的情况。在实际应用中,切片更常用,因为它提供了更大的灵活性和便利性。 总结: - 数组是长度固定的容器,切片是长度可变的容器; - 数组在声明时需要指定长度,切片则不需要; - 数组的长度一旦确定后就不能更改,而切片可以根据需要动态扩展或缩小; - 切片是基于数组的引用类型,可以通过指定起始索引和结束索引来指定子集。 参考资料: Golang中的「数组」和「切片」都是存储同一数据类型的容器,只不过Golang中的数组长度是固定的,而切片的长度是可变化的。 切片是引用类型,切片包装的数组称为该切片的底层数组。我们来看一段代码://a是一个数组,注意数组是一个固定长度的,初始化时候必须要指定长度,不指定长度的话就是切片了 a := int{1, 2, 3} //b是数组,是a...。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值