美文网首页Go语言GoGolang
Go 语言教程(4)——数据结构

Go 语言教程(4)——数据结构

作者: 浮x尘 | 来源:发表于2018-01-22 10:40 被阅读76次

    Array

    和其他语言的数组不同。

    • 数组是值类型,赋值和传参会复制整个数组,而不是指针
    • 数组长度必须是常量,且是类型的组成部分2[int]3[int] 是不同类型。
    • 支持 ==!= 操作符,因为内存总是被初始化过。
    • 指针数组 [n]*T、数组指针 *[n]T

    初始化

    a := [3]int{1, 2} // 未初始化元素值为 0
    b := [...]int{1, 2, 3, 4} // 通过初始化确定数组长度
    

    值拷贝会造成性能问题,通常会使用 slice,或数组指针。

    len、cap

    返回数组的长度与容量。

    a := [2]int{}
    println(len(a), cap(a)) // 2, 2
    

    Slice

    既然数组是值类型无法方便地传递给函数,肯定提供了方便的指针类型。

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

    slice 本质上是一个指向底层数组的结构体。

    struct Slice 
    {
        byte* array; // actual data
        uintgo len; // number of elements
        uintgo cap; // allocated number of elements
    }
    
    • 引用类型。但自身是结构体,值拷贝传递。
    • 每部分只有 8 个字节,长度永远不会超过 24 字节,在使用时不需要传递 silce 的地址,直接使用值传递。
    • 属性 len 表示可用元素数量,读写操作不能超过该限制。
    • 属性 cap 表⽰最⼤扩张容量,不能超出数组限制。
    • 如果 slice == nil,那么 len、cap 结果都等于 0
    data := [...]int{0, 1, 2, 3, 4, 5, 6}
    // len = high - low
    // cap = max - low
    slice := data[1:4:5] // [low : high : max]
    

    slice 读写操作实际目标是底层数组,需要注意索引号的区别。

    创建

    直接创建 slice 对象,自动分配底层数组。

    // 索引位置 8 的值为 100
    s1 := []int{0, 1, 2, 8:100} // 通过初始化表达式构造,创建时可使用索引号
    fmt.Println(s1, len(s1), cap(s1))
    
    s2 := make([]int, 6, 8) // 使用 make 创建,指定 len 和 cap 值
    fmt.Println(s2, len(s2), cap(s2))
    
    s3 := make([]int, 6) // 省略 cap,相当于 cap = len。
    fmt.Println(s3, len(s3), cap(s3))
    

    输出

    [0 1 2 3 0 0 0 0 100] 9 9
    [0 0 0 0 0 0] 6 8
    [0 0 0 0 0 0] 6 6
    

    使用 make 动态创建 slice,避免数组必须用常量做长度的麻烦。还可以用指针直接访问底层数组,变成普通数组操作。

    s := []int{0, 1, 2, 3}
    p := &s[2] // *int 获取底层数组元素指针
    
    fmt.Println(s) // [0 1 102 3]
    

    [][]T,是指元素类型为 []T

    reslice

    基于现有 slice 对象创建新 slcie 对象,以便在 cap 允许范围内调整属性。

    新对象依旧指向原底层数组

    s := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
    s1 := s[2:5] // [2 3 4]
    s2 := s1[2:6:7] // [4 5 6 7]
    s3 := s2[3:6] // Error
    

    append

    appendslice 尾部添加数据,返回新的 slice 对象。简单说就是在 array[slice.high] 后面写数据,会修改底层数组的值。

    data := [...]int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
    s := data[:3]
    
    s2 := append(s, 100, 200) // 添加多个值。
    s3 := append(s2, s...) // ... 解构赋值,相当于把 silce 展开
    
    fmt.Println(data)
    fmt.Println(s)
    fmt.Println(s2)
    fmt.Println(s3)
    

    输出

    [0 1 2 100 200 0 1 2 8 9]
    [0 1 2]
    [0 1 2 100 200]
    [0 1 2 100 200 0 1 2]
    

    追加后的容量一旦超过 slice.cap 的限制,会重新分配底层数组,即便原数组并未填满。

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

    remove

    官方并没有 remove 的相关接口,可以使用 append 变相实现该接口。

    // 删除第五个元素
    index := 5
    s := append(s[:index], s[index + 1:]...)
    

    insert

    官方也没有往指定位置插入元素的接口,依旧可以使用 append 实现。

    temp := append([]string{}, s[index:]...)
    s = append(s[:index], "insert")
    s = append(s, temp...)
    

    copy

    函数 copy 在两个 slice 之间复制数据,复制长度以 len 小的为准,复制较小的个数。两个 slice 允许指向同一底层数组,允许元素区间重叠。

    copy(to, from)

    data := [...]int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
    s := data[8:] // [8, 9]
    s2 := data[:5] // [0, 1, 2, 3, 4]
    copy(s2, s) // 从 s 复制到 s2
    fmt.Println(s2) // [8, 9, 2, 3, 4]
    fmt.Println(data) // 底层数组被改变 [8, 9, 2, 3, 4, 5, 6, 7, 8, 9]
    

    应及时将所需数据 copy 到较⼩的 slice,以便释放超⼤号底层数组内存。


    Map

    引用类型,Hash 表在任何语言中都有,C++ 中是 std::map<>Java 中是 Hashamp<>,在 Go 中则内置 map 不需要引入任何库。

    map 的键必须支持比较运算 == 和 !=,可以是 string、number、pointer、array、struct 等。值可以是任意类型,没有限制,取值的时候如果不存在就返回零值。

    map 本质上是一个字典指针

    type Map_K_V struct {
        // ...
    }
    
    type map[K]V struct {
        impl *Map_k_V
    }
    

    预先给 make 函数⼀个合理元素数量参数,有助于提升性能。因为事先申请⼀⼤块内存,可避免后续操作时频繁扩张。

    m := make(map[string]int, 1000)

    常见操作

    m := map[string]int{
        "a": 1,
    }
    
    // 判断 key 是否存在
    if v, ok := m["a"]; ok {
        println(v)
    }
    
    
    // 不存在的 key 返回零值
    m["b"] // 0
    
    // 新增或修改
    m["b"] = 2
    
    // 删除,如果 key 不存在不会出错
    delete(map, key)
    
    // 获取键值对数量
    println(len(m))
    
    // 迭代,随机顺序返回
    // range 返回 value 的临时拷贝
    for k, v := range m {
        println(k, v)
    }
    

    map 中取回的是一个 value 临时复制品,对其成员修改不会改变源对象。

    正确的做法是完整对象替换或使用指针作为 value

    Struct

    值语义,赋值和传参会复制全部内容。可⽤ "_" 定义补位字段,支持指向自身类型的指针成员。

    支持 ==、!= 操作符。

    type Node struct {
        _ int
        id int
        data *byte
        next *Node
    }
    
    type User struct {
        id int
        name string
    }
    
    user1 := User{1, "Tom"}
    user2 := User{1, "Tom"}
    
    fmt.Println(user1 == user2) // true
    

    空结构 "节省" 内存,⽐如⽤来实现 set 数据结构,或者实现没有 "状态" 只有⽅法的 "静态类"。

    var null struct{}
    
    set := make(map[string]struct{})
    set["a"] = null
    

    标签

    可以定义标签,用反射读取。

    匿名字段

    匿名字段本质上是一种语法糖,只是一个与成员类型同名,且不包含包名的字段。被匿名嵌入的可以是任何类型,也包括指针。

    type User struct {
        name string
    }
    
    type Manager struct {
        User // 匿名字段
        title string
    }
    
    m := Manager{
        User: User{"Tom"}, // 匿名字段的显式字段名,和类型名相同。
        title: "Administrator",
    }
    
    m.name = "jack" // 访问匿名字段成员
    

    可以像访问普通字段一样访问匿名字段成员,编译器从外向内逐级查找所有层次的匿名字段,直到发现目标或出错。

    外层同名字段会遮蔽嵌⼊字段成员,相同层次的同名字段也会让编译器⽆所适从。解决⽅法是使⽤显式字段名。

    本质上就是不能有歧义,不能同时找到两个,而不知道使用哪一个。

    面向对象

    面向对象三大特征里,Go 仅支持封装,尽管匿名字段的内存布局和行为类似继承。没有 class 关键字,没有继承、多态等等。

    内存布局和 C struct 相同,没有任何附加的 object 信息。

    type User struct {
        id int
        name string
    }
    
    type Manager struct {
        User
        title string
    }
    
    m := Manager{User{1, "Tom"}, "Administrator"}
    
    // var u User = m // Error: cannot use m (type Manager) as type User in assignment. 没有继承自然没有多态
    
    var u User = m.User // 同类型拷贝
    

    内存布局

        |<-------- User:24 ------->|<-- title:16 -->|
        +-------------+------------+----------------+              +---------------+
     m  |      1      |    string  |     string     |              | Administrator | [n]byte 
        +-------------+------------+----------------+              +---------------+
                            |               |                              |
                            |               +--->>>------------->>>--------+
                            |
                            +--->>>----------------------------------------+
                                                                           |
                            +--->>>-------------------------------------+  |
                            |                                           |  |
        +-------------+------------+                                 +-------------+
     u  |     1       |    string  |                                 |    Tom      |  [n]byte
        +-------------+------------+                                 +-------------+                                  |<-   id:8  ->|<- name:16->|                                      
    

    相关文章

      网友评论

        本文标题:Go 语言教程(4)——数据结构

        本文链接:https://www.haomeiwen.com/subject/idqlaxtx.html