GoLang 切片

Golang Series

Hello GoLang: https://blog.yexca.net/archives/154
GoLang (var and const) 变量与常量: https://blog.yexca.net/archives/155
GoLang (func) 函数: https://blog.yexca.net/archives/156
GoLang (slice and map) 切片: This Page
GoLang (OOP) 面向对象: https://blog.yexca.net/archives/162
GoLang (reflect) 反射: https://blog.yexca.net/archives/204
GoLang (struct tag) 结构体标签: https://blog.yexca.net/archives/205
GoLang (goroutine) go 程: https://blog.yexca.net/archives/206
GoLang (channel) 通道: https://blog.yexca.net/archives/207


Go 的切片是对数组的抽象

数组

数组的长度不可改变

package main

import "fmt"

func main() {
    // 定义方式一
	var arr1 [10]int
    // 遍历
	for i := 0; i < len(arr1); i++ {
		fmt.Println("arr1[", i, "]:", arr1[i])
	}

    // 定义方式二,赋值
	arr2 := [10]int{0, 1, 2, 3}
    // range遍历
	for index, value := range arr2 {
		fmt.Println("index =", index, "value =", value)
	}

    // 定义不同长度
	var arr3 [4]int

	fmt.Printf("type of arr1 is %T\n", arr1) // [10]int
	fmt.Printf("type of arr1 is %T\n", arr2) // [10]int
	fmt.Printf("type of arr1 is %T", arr3)	 // [4]int
}

编译运行后可以发现 arr3 与 arr1,arr2 类型不同,那么在定义函数形参时也需要指定相应类型

func test(arr [4]int) {
	for i := 0; i < len(arr); i++ {
		fmt.Println("fmt_arr[", i, "]:", arr[i])
	}
}

上述函数只能传递 arr3,值传递,修改值不影响原数据

定义切片

与数组相比切片长度不固定,可追加元素 (动态数组),在追加时可能使切片的容量增大

定义切片可以通过声明一个未指定大小的数组

var name []type
// 例如
var s []int

或者使用 make() 函数来创建切片

var slice []type = make([]type, len) // len 为切片初始长度
// 也可以简写为
slice := make([]type, len)

可以使用可选参数 capacity 指定容量,省略与 length 相同

var slice []type = make([]type, length, capacity)

切片初始化

直接初始化

s := []int {1, 2, 3}

将数组值为切片初始化,从 startIndex 到 endIndex-1,这俩值都可省略

s := arr[startIndex:endIndex]

省略 startIndex 或 endIndex 表示从第一个元素开始索引或索引到最后一个元素

len() and cap()

切片是可以索引的,通过 len() 函数获取长度

而 cap() 为计算容量的方法,可以测量切片最长可以达到多少

package main

import "fmt"

// 切片传递为引用传递,函数内修改影响原数据
func printSlice(slice []int) {
	fmt.Printf("len=%d, cap=%d, slice=%v", len(slice), cap(slice), slice)
}

func main() {
	s := make([]int, 3, 5)
	printSlice(s)
}

/*
 * 输出
 * len=3, cap=5, slice=[0 0 0]
 */

空切片

一个切片在未初始化之前默认为 nil (空切片),长度为 0

package main

import "fmt"

func printSlice(slice []int) {
	fmt.Printf("len=%d, cap=%d, slice=%v\n", len(slice), cap(slice), slice)
}

func main() {
	var s []int
	printSlice(s)
    // 判断是否为空
	if s == nil {
		fmt.Println("slice is empty")
	}
}

切片截取

通过设置上下限截取切片

package main

import "fmt"

func printSlice(slice []int) {
	fmt.Printf("len=%d, cap=%d, slice=%v\n", len(slice), cap(slice), slice)
}

func main() {
	s := []int{0, 1, 2, 3, 4, 5, 6, 7}

    // 打印原始切片
    fmt.Println(s)
    
    // 从2(包含)到5(不包含)
	printSlice(s[2:5])
    // 从第一个到5(不包含)
	printSlice(s[:5])
    // 从第二个到最后一个
	printSlice(s[2:])

    // 这样赋值修改 subS 将影响到 s
	subS := s[1:6]
	printSlice(subS)
}

append() and copy()

增加切片的容量与拷贝切片

package main

import "fmt"

func printSlice(slice []int) {
	fmt.Printf("len=%d, cap=%d, slice=%v\n", len(slice), cap(slice), slice)
}

func main() {
	var s []int
	printSlice(s)

    // 增加一个元素
	s = append(s, 0)
	printSlice(s)

    // 增加多个元素
	s = append(s, 1, 2, 3, 4)
	printSlice(s)

    // 创建一个容量为 s 两倍的 s2
	s2 := make([]int, len(s), cap(s)*2)
    // 拷贝 s 到 s2,此时修改 s2 不影响 s
	copy(s2, s)
	printSlice(s2)
}

切片的扩容:如果追加值超过容量,则将容量增加两倍

map

map 有两种声明方式

package main

import "fmt"

func main() {
	var map1 = make(map[string]string)
	// 插入数据
	map1["one"] = "1"
	map1["two"] = "2"
	fmt.Println(map1) // map[one:1 two:2]
}

第二种

package main

import "fmt"

func main() {
	map1 := map[string]string{
		"one": "1",
		"two": "2",
	}
	fmt.Println(map1)
}

map 嵌套

package main

import "fmt"

func main() {
	map1 := make(map[string]map[string]string)
	map1["first"] = make(map[string]string, 2)
	map1["first"]["one"] = "1"
	map1["first"]["two"] = "2"
	fmt.Println(map1)
}

/*
 * 输出
 * map[first:map[one:1 two:2]]
 */

修改遍历与删除

package main

import "fmt"

func main() {
	map1 := make(map[string]map[string]string)
	map1["first"] = make(map[string]string, 2)
	map1["first"]["one"] = "1"
	map1["first"]["two"] = "2"
	// 修改
	map1["first"]["one"] = "one"
	fmt.Println(map1)
    //遍历
    for key, value := range map1{
        fmt.println("key =", key, "value =", value)
    }
	// 删除
	delete(map1, "first")
	fmt.Println(map1)
}

判断是否有某值

package main

import "fmt"

func main() {
	map1 := make(map[string]string)
	map1["one"] = "1"
	val, key := map1["one"]
	if key {
		fmt.Println(val)
	} else {
		fmt.Println("empty")
	}
}
This post is licensed under CC BY-NC-SA 4.0 by the author.