网站首页 > 文章中心 > 其它

go语言结构体切片

作者:小编 更新时间:2023-08-11 21:49:18 浏览量:240人看过

Go切片数组深度解析

Go 中的分片数组,实际上有点类似于Java中的ArrayList,是一个可以扩展的数组,但是Go中的切片由比较灵活,它和数组很像,也是基于数组,所以在了解Go切片前我们先了解下数组.

数组简单描述就由相同类型元素组成的数据结构, 在创建初期就确定了长度,是不可变的.

但是Go的数组类型又和C与Java的数组类型不一样, NewArray 用于创建一个数组,从源码中可以看出最后返回的是 Array{}的指针,并不是第一个元素的指针,在Go中数组属于值类型,在进行传递时,采取的是值传递,通过拷贝整个数组.Go语言的数组是一种有序的struct.

Go 语言的数组有两种不同的创建方式,一种是显示的初始化,一种是隐式的初始化.

go语言结构体切片-图1

注意一定是使用 [...]T 进行创建,使用三个点的隐式创建,编译器会对数组的大小进行推导,只是Go提供的一种语法糖.

Go中的数组属于值类型,通常应该存储于栈中,局部变量依然会根据逃逸分析确定存储栈还是堆中.

编译器对数组函数中做两种不同的优化:

在静态区完成赋值后复制到栈中.

由于数组是值类型,那么赋值和函数传参操作都会复制整个数组数据.

不管是赋值或函数传参,地址都不一致,发生了拷贝.如果数组的数据较大,则会消耗掉大量内存.那么为了减少拷贝我们可以主动的传递指针呀.

go语言结构体切片-图2

地址是一样的,不过传指针会有一个弊端,从打印结果可以看到,指针地址都是同一个,万一原数组的指针指向更改了,那么函数里面的指针指向都会跟着更改.

同样的我们将数组转换为切片,通过传递切片,地址是不一样的,数组值相同.

切片是引用传递,所以它们不需要使用额外的内存并且比使用数组更有效率.

所以,切片属于引用类型.

通过这种方式可以将数组转换为切片.

中间不加三个点就是切片,使用这种方式创建切片,实际上是先创建数组,然后再通过第一种方式创建.

使用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语言 排序与搜索切片

Go语言标准库中提供了sort包对整型,浮点型,字符串型切片进行排序,检查一个切片是否排好序,使用二分法搜索函数在一个有序切片中搜索一个元素等功能.

关于sort包内的函数说明与使用,请查看

今天这一节简单讲几个sort包中常用的函数

在Go语言中,对字符串的排序都是按照字节排序,也就是说在对字符串排序时是区分大小写的.

二分搜索算法

Go语言中提供了一个使用二分搜索算法的sort.Search(size,fn)方法:每次只需要比较㏒?n个元素,其中n为切片中元素的总数.

sort.Search(size,fn)函数接受两个参数:所处理的切片的长度和一个将目标元素与有序切片的元素相比较的函数,该函数是一个闭包,如果该有序切片是升序排列,那么在判断时使用 有序切片的元素 = 目标元素.该函数返回一个int值,表示与目标元素相同的切片元素的索引.

在切片中查找出某个与目标字符串相同的元素索引

go切片遍历

对于切片的顺序遍历,一般使用 range 就可以了.

这里有一个问题需要注意一下,如果这里的切片nums不是基本数据类型而是结构体.range遍历出来的value值是拷贝值而并非原结构体,修改value中的值不会改变原切片中的值.如果要遍历修改,可以将切片的结构体改为指针,或都索引来取值.

一般情况下逆序遍历思路就是for size-1到0.

二般的也可以使用range来遍历

golang函数返回slice和返回 slice的指针有什么区别

按照你的定义,slice是切片,而p是指针.切片是一个结构体头部+数组区域,其头部结构定义如下:

struct Slice

{ // must not move anything

byte* array; // actual data

uintgo len; // number of elements

uintgo cap; // allocated number of elements

};

所以呢,slice的返回其实是头部值返回,函数内外的地址是不同的,这也导致主程序中,ss与pp不同.因为ss是新分配的,pp则是与子程序testInterface中的slice相同.简单修改你的代码,通过输出对比,会非常清晰:

package main

import (

"fmt"

go语言结构体切片-图3

)

func testInterface() (slice interface{}, p interface{}) {

slice = make([]int, 10)

p = slice

fmt.Println("debug:testInterface")

fmt.Println(slice)//两个地址应该相同

fmt.Println(p) //两个地址应该相同

return slice, p

}

func main() {

fmt.Println("debug:main")

ss, pp := testInterface()

fmt.Println(ss)

fmt.Println(pp) //应该与子程序的输出一致

另外,第一个问题就不用多解释,依然是值和指针不同了.

以上就是土嘎嘎小编为大家整理的go语言结构体切片相关主题介绍,如果您觉得小编更新的文章只要能对粉丝们有用,就是我们最大的鼓励和动力,不要忘记讲本站分享给您身边的朋友哦!!

版权声明:倡导尊重与保护知识产权。未经许可,任何人不得复制、转载、或以其他方式使用本站《原创》内容,违者将追究其法律责任。本站文章内容,部分图片来源于网络,如有侵权,请联系我们修改或者删除处理。

编辑推荐

热门文章