Go 中数组的长度是不可改变的,而 Slice 解决的就是对不定长数组的需求.他们的区别主要有两点.
数组:
切片:
注意 1
虽然数组在初始化时也可以不指定长度,但 Go 语言会根据数组中元素个数自动设置数组长度,并且不可改变.切片通过 append 方法增加元素:
如果将 append 用在数组上,你将会收到报错:first argument to append must be slice.
切片不只有长度(len)的概念,同时还有容量(cap)的概念.所以呢切片其实还有一个指定长度和容量的初始化方式:
此外,切片还可以从一个数组中初始化(可应用于如何将数组转换成切片):
上述例子通过数组 a 初始化了一个切片 s.
当切片和数组作为参数在函数(func)中传递时,数组传递的是值,而切片传递的是指针.所以呢当传入的切片在函数中被改变时,函数外的切片也会同时改变.相同的情况,函数外的数组则不会发生任何变化.
、数组
与其他大多数语言类似,Go语言的数组也是一个元素类型相同的定长的序列.
(1)数组的创建.
复制代码代码如下:
fmt.Println(iarray1)
}
结果:
[0 0 0 0 0]
第二段:切片
Go语言中,切片是长度可变、容量固定的相同的元素序列.Go语言的切片本质是一个数组.容量固定是因为数组的长度是固定的,切片的容量即隐藏数组的长度.长度可变指的是在数组长度的范围内可变.
(1)切片的创建.
①.)make ( []Type ,length, capacity )
fmt.Println(slice1)
输出为:
[0 0 0 0 0 0 0 0 0]
[]
slice0 := []string{"a", "b", "c", "d", "e"}
[a b c d e] [c d] [a b c]
func test10() {
fmt.Println("\n~~~~~~元素遍历~~~~~~")
for _, ele := range slice0 {
fmt.Print(ele, " ")
fmt.Println("\n~~~~~~索引遍历~~~~~~")
for index := range slice0 {
fmt.Print(slice0[index], " ")
fmt.Println("\n~~~~~~元素索引共同使用~~~~~~")
for index, ele := range slice0 {
fmt.Print(ele, slice0[index], " ")
fmt.Println("\n~~~~~~修改~~~~~~")
fmt.Println(slice0)
只有一个元素时,该元素代表索引.
结果为:
~~~~~~元素遍历~~~~~~
a b c d e
~~~~~~索引遍历~~~~~~
~~~~~~元素索引共同使用~~~~~~
aa bb cc dd ee
~~~~~~修改~~~~~~
func test11() {
fmt.Printf("slice的长度为:%d,slice为:%v\n", len(slice), slice)
fmt.Printf("追加后,slice的长度为:%d,slice为:%v\n", len(slice), slice)
fmt.Printf("slicecp的长度为:%d,slicecp为:%v\n", len(slicecp), slicecp)
copy(slicecp, slice)
fmt.Printf("复制赋值后,slicecp的长度为:%d,slicecp为:%v\n", len(slicecp), slicecp)
追加、复制切片,用的是内置函数append和copy,copy函数返回的是最后所复制的元素的数量.
内置函数append可以向一个切片后追加一个或多个同类型的其他值.如果追加的元素数量超过了原切片容量,那么最后返回的是一个全新数组中的全新切片.如果没有超过,那么最后返回的是原数组中的全新切片.无论如何,append对原切片无任何影响.如下示例:
fmt.Printf("slice为:%v\n", slice)
Go 中的分片数组,实际上有点类似于Java中的ArrayList,是一个可以扩展的数组,但是Go中的切片由比较灵活,它和数组很像,也是基于数组,所以在了解Go切片前我们先了解下数组.
数组简单描述就由相同类型元素组成的数据结构, 在创建初期就确定了长度,是不可变的.
但是Go的数组类型又和C与Java的数组类型不一样, NewArray 用于创建一个数组,从源码中可以看出最后返回的是 Array{}的指针,并不是第一个元素的指针,在Go中数组属于值类型,在进行传递时,采取的是值传递,通过拷贝整个数组.Go语言的数组是一种有序的struct.
Go 语言的数组有两种不同的创建方式,一种是显示的初始化,一种是隐式的初始化.
注意一定是使用 [...]T 进行创建,使用三个点的隐式创建,编译器会对数组的大小进行推导,只是Go提供的一种语法糖.
Go中的数组属于值类型,通常应该存储于栈中,局部变量依然会根据逃逸分析确定存储栈还是堆中.
编译器对数组函数中做两种不同的优化:
在静态区完成赋值后复制到栈中.
由于数组是值类型,那么赋值和函数传参操作都会复制整个数组数据.
不管是赋值或函数传参,地址都不一致,发生了拷贝.如果数组的数据较大,则会消耗掉大量内存.那么为了减少拷贝我们可以主动的传递指针呀.
地址是一样的,不过传指针会有一个弊端,从打印结果可以看到,指针地址都是同一个,万一原数组的指针指向更改了,那么函数里面的指针指向都会跟着更改.
同样的我们将数组转换为切片,通过传递切片,地址是不一样的,数组值相同.
切片是引用传递,所以它们不需要使用额外的内存并且比使用数组更有效率.
所以,切片属于引用类型.
通过这种方式可以将数组转换为切片.
中间不加三个点就是切片,使用这种方式创建切片,实际上是先创建数组,然后再通过第一种方式创建.
使用make创建切片,就不光编译期了,make创建切片会涉及到运行期.1. 切片的大小和容量是否足够小;
切片是否发生了逃逸,最终在堆上初始化.如果切片小的话会先在栈或静态区进行创建.
切片有一个数组的指针,len是指切片的长度, cap指的是切片的容量.
cap是在初始化切片是生成的容量.
发现切片的结构体是数组的地址指针array unsafe.Pointer,而Go中数组的地址代表数组结构体的地址.
slice 中得到一块内存地址,array[0]或者unsafe.Pointer(array[0]).
也可以通过地址构造切片
nil切片:指的unsafe.Pointer 为nil
空切片:
创建的指针不为空,len和cap为空
当一个切片的容量满了,就需要扩容了.怎么扩,策略是什么?
如果原来数组切片的容量已经达到了最大值,再想扩容, Go 默认会先开一片内存区域,把原来的值拷贝过来,然后再执行 append() 操作.这种情况对现数组的地址和原数组地址不相同.
从上面结果我们可以看到,如果用 range 的方式去遍历一个切片,拿到的 Value 其实是切片里面的值拷贝,即浅拷贝.所以每次打印 Value 的地址都不变.
由于 Value 是值拷贝的,并非引用传递,所以直接改 Value 是达不到更改原切片值的目的的,需要通过 slice[index] 获取真实的地址.
以上就是土嘎嘎小编为大家整理的go语言数组与切片相关主题介绍,如果您觉得小编更新的文章只要能对粉丝们有用,就是我们最大的鼓励和动力,不要忘记讲本站分享给您身边的朋友哦!!