当前位置: 首页 > news >正文

Go基础编程 - 05 - 数组与切片

目录

    • 1. 数组
    • 2. 切片
      • 2.1. slice 声明、初始化
      • 2.2. slice 操作
      • 2.3. append() 追加切片、扩容
      • 2.4. 字符串和切片
    • 3. Copy
    • 4. Array、Slice 内存布局

上一篇:基本类型、常量和变量

下一篇:指针


1. 数组

数组是同一种类型固定长度的序列(有长度、类型构成)。一但定义,长度不可改变。

  • 数组为值类型,赋值和传参会复制整个数组,而不是指针。
  • 数组声明时,必须固定长度;长度是数组类型的一部分,不同长度的数组类型表示不同的类型([2]int[3]int 为不同的数组类型);
  • 数组未赋值,则初始化为类型缺省值(零值)。
  • 内置函数lencap都返回数组长度。
  • 越界访问产生panic。
  • 指针数组:[n]*T,数组指针:*[n]T
  • 值拷贝行为会造成性能问题,通常会建议使用 slice,或数组指针。
package mainimport ("fmt""reflect"
)func main() {// 初始化var arr1 [5]intvar arr2 = [5]int{0, 1, 2, 3, 4}var arr3 = [3]int{0, 2: 4}var arr4 = [...]int{1, 2, 3}fmt.Println(arr1, arr2, arr3, arr4)// 长度是数组类型的一部分,不同长度的数组类型为不同类型if reflect.TypeOf(arr1) == reflect.TypeOf(arr2) {fmt.Println("arr1 和 arr2 数据类型相同!")}if reflect.TypeOf(arr1) != reflect.TypeOf(arr3) {fmt.Println("arr1 和 arr3 数据类型不相同!")}// 数组为值类型:赋值和传参复制整个数组arr1copy := arr1arr1copy[0] = 10fmt.Println(arr1, arr1copy)// 值拷贝行为会造成性能问题,通常会建议使用 slice,或数组指针。fn := func(arr [5]int) {arr[1] = 20}fn(arr1)fmt.Println(arr1)// 指针数组:[n]*Tvar pa = [2]*int{new(int)}*pa[0] = 10pa[1] = new(int) // 引用类型必须先分配内存,须使用new分配内存*pa[1] = 20      // 若不分配内存(注释上一行代码),则产生panic: runtime error: invalid memory address or nil pointer dereferencefmt.Printf("pa:%v  \n", pa)// 数组指针:*[n]Tvar ap *[3]int // var ap = new([3]int)ap = new([3]int)*ap = [3]int{4, 5, 6}fmt.Printf("ap:%v  psr:%p  type:%T  \n", *ap, ap, ap)// 多维数组multArr := [...][2]int{{0, 0}, {1, 1}, {2, 2}}fmt.Println(multArr)
}

2. 切片

切片并不是数组或数组指针。它通过内部指针和相关属性引用数组片段,以实现变长方案。

  • 切片是引用类型;底层数据结构是一个数组,可以看作是对数组某个连续片段的引用。
  • 切片是可变长的(包含长度、容量)。
  • 长度:用len()函数求长度,表示可用元素数量,读写操作不能超过该限制。
  • 容量:用cap()函数求最大容量,为切片底层数组的长度。
  • 扩容及规律: 扩容通常以 2 倍容量重新分配底层数组;当原有切片容量大于 1024 时,以 1.25 倍扩容。
  • 若 slice == nil,则 lencap结果都等于 0。

2.1. slice 声明、初始化

package mainimport "fmt"func main() {// 声明var s []int     if s == nil {fmt.Println("s is nil")}s1 := []int{1, 2, 3, 4}// 使用 make()var s2 []int = make([]int, 0)s3 := make([]int, 5)s4 := make([]int, 5, 8)fmt.Println(s, s1, s2, s3, s4)fmt.Println("Cap:", cap(s), cap(s1), cap(s2), cap(s3), cap(s4))// 从数组切片arr := [5]int{1, 2, 3, 4, 5}var s5 []ints5 = arr[1:4] // 前包后不包fmt.Println(s5, len(s5), cap(s5))
}

2.2. slice 操作

  • slice 的索引起点为 0。
  • 切片截取:s[i:j:max],满足 i <= j <= max;其中,i为起始索引(包含),省略值 0j结束索引(不包含),省略值len(s)max为截取最大索引,省略值len(s)
  • 截取新切片长度 = j - i
  • 截取新切片容量 = max - i
  • 截取新切片开始索引非 0,指针地址指向底层数组中新切片开始索引所在的地址;起始地址相同的slice指针地址相同,例:s[i], s[i:], s[i:j], s[i:j:max] 的指针引用地址都相同。
package mainimport "fmt"func main() {s := [...]int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20}v := s[0]fmt.Println(v)s1 := s[2:5]s2 := s[2:8]s3 := s[:10:12]fmt.Println("val", v, s1, s2, s3)fmt.Println("len", len(s1), len(s2), len(s3))fmt.Println("cap", cap(s1), cap(s2), cap(s3))fmt.Printf("prs %p %p %p \n", s1, s2, s3)
}

2.3. append() 追加切片、扩容

append() 向 slice 尾部添加数据,返回新的 slice 对象。

扩容通常以 2 倍容量重新分配底层数组。在大批量添加数据时,建议一次性分配足够大的空间,以减少内存分配和数据复制开销。或初始化足够长的 len 属性,改用索引号进行操作。及时释放不再使用的 slice 对象,避免持有过期数组,造成 GC 无法回收。

package mainimport "fmt"func main() {arr := []int{0, 1, 2, 3, 4}fmt.Println("len =", len(arr), "cap =", cap(arr))       // len = 5  cap = 5arr1 := append(arr, 5)       // 追加数据,超出原切片容量,则进行扩容fmt.Println("len =", len(arr1), "cap =", cap(arr1))     // len = 6  cap = 10fmt.Printf("prs: %p %p \n", arr, arr1)  // 对比底层数组起始指针,扩容后重新分配底层数组,与原数组无关。arr1 = append(arr1, []int{6, 7, 8, 9, 10}...)fmt.Println("len =", len(arr1), "cap =", cap(arr1))     // len = 11 cap = 20
}

2.4. 字符串和切片

string 底层就是一个 byte/rune 的数组,因此,也可以进行切片操作。

package mainimport "fmt"func main() {str := "Hello, Chain"s1 := str[:5]println(s1) // Hello//string 本身是不可变的,因此要改变string中字符;需将字符转为slice,进行更改,再转为string//str[0] = 'X'     // 报错:cannot assign to str[0] (value of type byte)s := []byte(str) // 英文字符使用 byte,代表ASCII码的一个字符// 赋值需要使用单引号(''),单引号定义一个字符,双引号定义个字符串s = append(s, '!')str = string(s)fmt.Println(str)str2 := "你好,中国"s2 := []rune(str2) // 包含中文字符使用 rune,代表一个UTF-8字符s2[3] = '伟's2[4] = '大's2 = append(s2, []rune{'的', '中', '国'}...)fmt.Println(string(s2))
}

3. Copy

  • 深拷贝:拷贝的是数据本身;值类型数据,默认是深拷贝,Array、Int、Sting、Struct、Bool、Float。
  • 浅拷贝:拷贝的是数据地址;引用类型数据,默认浅拷贝,Map、Slice。
package mainimport "fmt"func main() {arr := [...]int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}arr2 := arrarr[1] = 10fmt.Println(arr, arr2)fmt.Printf("prs: %p %p \n", &arr, &arr2) // Array 属于深copyfmt.Println()s1 := arr[:6]s2 := arr[4:]s1[5] = 50 // Slice 属于浅copyfmt.Println(s1, s2)fmt.Printf("prs:%p %p \n", s1, s2)// copy函数只能用于切片,属于浅拷贝;允许元素区间重叠copy(s1, s2)s1[1] = 100fmt.Println(s1, s2, arr)
}

4. Array、Slice 内存布局


array-slice

相关文章:

  • (一)SvelteKit教程:hello world
  • windows桌面运维----第三天
  • CCAA质量管理【学习笔记】​​ 备考知识点笔记(三)质量管理方法与常见工具
  • 了解并解决 Flutter 中的灰屏问题
  • 瞬间将模型改为原来的60-200倍小
  • PHP框架详解 - CakePHP框架
  • 细说MCU输出互补型PWM波形时设置死区时间的作用
  • 大数据之Hadoop的特点是什么?有什么优缺点?有哪些发行版本?
  • 军用FPGA软件 Verilog语言的编码准测之触发器、锁存器
  • 各类存储器类型(RAM、ROM、FLASH、DRAM、SRAM)
  • Kafka之ISR机制的理解
  • Java程序设计语言的特点
  • 【Quartus 13.0】NIOS II 部署UART 和 PWM
  • phpStudy里面的MySQL启动不了
  • 这些已经死去的软件,依旧无可替代
  • python3.6+scrapy+mysql 爬虫实战
  • C++类的相互关联
  • iOS | NSProxy
  • Javascript编码规范
  • JavaWeb(学习笔记二)
  • Python学习笔记 字符串拼接
  • Ruby 2.x 源代码分析:扩展 概述
  • ⭐ Unity + OpenCV 实现实时图像识别与叠加效果
  • Vim 折腾记
  • 给新手的新浪微博 SDK 集成教程【一】
  • 爬虫模拟登陆 SegmentFault
  • 使用parted解决大于2T的磁盘分区
  • 栈实现走出迷宫(C++)
  • 你学不懂C语言,是因为不懂编写C程序的7个步骤 ...
  • 如何通过报表单元格右键控制报表跳转到不同链接地址 ...
  • #QT(一种朴素的计算器实现方法)
  • #图像处理
  • (1) caustics\
  • (26)4.7 字符函数和字符串函数
  • (Java数据结构)ArrayList
  • (vue)页面文件上传获取:action地址
  • (Windows环境)FFMPEG编译,包含编译x264以及x265
  • (备忘)Java Map 遍历
  • (黑马出品_高级篇_01)SpringCloud+RabbitMQ+Docker+Redis+搜索+分布式
  • (每日一问)基础知识:堆与栈的区别
  • (四)JPA - JQPL 实现增删改查
  • (四)软件性能测试
  • (转) ns2/nam与nam实现相关的文件
  • (转)Spring4.2.5+Hibernate4.3.11+Struts1.3.8集成方案一
  • (转)菜鸟学数据库(三)——存储过程
  • (转)如何上传第三方jar包至Maven私服让maven项目可以使用第三方jar包
  • (轉)JSON.stringify 语法实例讲解
  • (自用)交互协议设计——protobuf序列化
  • ..回顾17,展望18
  • .apk 成为历史!
  • .Net MVC4 上传大文件,并保存表单
  • .NET 常见的偏门问题
  • .net访问oracle数据库性能问题
  • .net和jar包windows服务部署
  • .pyc文件还原.py文件_Python什么情况下会生成pyc文件?