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

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->|                                      
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,772评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,458评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,610评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,640评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,657评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,590评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,962评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,631评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,870评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,611评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,704评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,386评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,969评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,944评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,179评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,742评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,440评论 2 342

推荐阅读更多精彩内容

  • 出处---Go编程语言 欢迎来到 Go 编程语言指南。本指南涵盖了该语言的大部分重要特性 Go 语言的交互式简介,...
    Tuberose阅读 18,396评论 1 46
  • Go语言做Web编程非常方便,并且在开发效率和程序运行效率方面都非常优秀。相比于Java,其最大的优势就是简便易用...
    暗黑破坏球嘿哈阅读 8,986评论 6 67
  • Go入门 Go介绍 部落图鉴之Go:爹好还这么努力? 环境配置 安装 下载源码编译安装 下载相应平台的安装包安装 ...
    齐天大圣李圣杰阅读 4,567评论 0 26
  • 刚从哈尔滨回来。去了趟北方,整个人感觉都不好了。什么事情都要找人,没一样是按照规矩办事的。简直都懒得发牢骚。嘴巴上...
    秦潇越阅读 211评论 0 1
  • 题目:口才方面的“渔王” 关键词:渔王 手把手教 少走弯路 教训 统计:总结10分钟,120分钟练习30遍, 练习...
    008一字千金537践行思想阅读 187评论 0 0