Go 专栏|复合数据类型:字典 map 和 结构体 struct

原文链接: Go 专栏|复合数据类型:字典 map 和 结构体 struct

楼下新开了一家重庆砂锅肥肠,扩音喇叭一直在放:正宗的老重庆砂锅肥肠,麻辣可口,老巴适了。

正不正宗不知道,反正听口音,我以为我回东北了。

本篇介绍复合数据类型的最后一篇:字典和结构体。内容很重要,编程时用的也多,需要熟练掌握才行。

本文所有代码基于 go1.16.6 编写。

字典

字典是一种非常常用的数据结构,Go 中用关键词 map 表示,类型是 map[K]VKV 分别是字典的键和值的数据类型,其中键必须支持相等运算符,比如数字,字符串等。

创建字典

有两种方式可以创建字典,第一种是直接使用字面量创建;第二种使用内置函数 make

字面量方式创建:

// 字面量方式创建
var m = map[string]int{"a": 1, "b": 2}
fmt.Println(m) // map[a:1 b:2]

使用 make 创建:

// 使用 make 创建
m1 := make(map[string]int)
fmt.Println(m1)

还可以初始化字典的长度。在已知字典长度的情况下,直接指定长度可以提升程序的执行效率。

// 指定长度
m2 := make(map[string]int, 10)
fmt.Println(m2)

字典的零值是 nil,对值是 nil 的字典赋值会报错。

// 零值是 nil
var m3 map[string]int
fmt.Println(m3 == nil, len(m3) == 0) // true true
// nil 赋值报错
// m3["a"] = 1
// fmt.Println(m3)  // panic: assignment to entry in nil map

使用字典

赋值:

// 赋值
m["c"] = 3
m["d"] = 4
fmt.Println(m) // map[a:1 b:2 c:3 d:4]

取值:

// 取值
fmt.Println(m["a"], m["d"]) // 1 4
fmt.Println(m["k"])         // 0

即使在 Key 不存在的情况下,也是不报错的。而是返回对应类型的零值。

删除元素:

// 删除
delete(m, "c")
delete(m, "f") // key 不存在也不报错
fmt.Println(m) // map[a:1 b:2 d:4]

获取长度:

// 获取长度
fmt.Println(len(m)) // 3

判断键是否存在:

// 判断键是否存在
if value, ok := m["d"]; ok {
    fmt.Println(value) // 4
}

和 Python 对比起来看,这个用起来就很爽。

遍历:

// 遍历
for k, v := range m {
    fmt.Println(k, v)
}

引用类型

map 是引用类型,所以在函数间传递时,也不会制造一个映射的副本,这点和切片类似,都很高效。

package main

import "fmt"

func main() {
    ...

    // 传参
    modify(m)
    fmt.Println("main: ", m) // main:  map[a:1 b:2 d:4 e:10]
}

func modify(a map[string]int) {
    a["e"] = 10
    fmt.Println("modify: ", a) //   modify:  map[a:1 b:2 d:4 e:10]
}

结构体

结构体是一种聚合类型,包含零个或多个任意类型的命名变量,每个变量叫做结构体的成员。

创建结构体

首先使用 type 来自定义一个结构体类型 user,里面有两个成员变量,分别是:nameage

// 声明结构体
type user struct {
    name string
    age  int
}

结构体的初始化有两种方式:

第一种是按照声明字段的顺序逐个赋值,这里需要注意,字段的顺序要严格一致。

// 初始化
u1 := user{"zhangsan", 18}
fmt.Println(u1) // {zhangsan 18}

这样做的缺点很明显,如果字段顺便变了,那么凡是涉及到这个结构初始化的部分都要跟着变。

所以,更推荐使用第二种方式,按照字段名字来初始化。

// 更好的方式
// u := user{
//  age: 20,
// }
// fmt.Println(u)   // { 20}
u := user{
    name: "zhangsan",
    age:  18,
}
fmt.Println(u) // {zhangsan 18}

未初始化的字段会赋值相应类型的零值。

使用结构体

使用点号 . 来访问和赋值成员变量。

// 访问结构体成员
fmt.Println(u.name, u.age) // zhangsan 18
u.name = "lisi"
fmt.Println(u.name, u.age) // lisi 18

如果结构体的成员变量是可比较的,那么结构体也是可比较的。

// 结构体比较
u2 := user{
    age:  18,
    name: "zhangsan",
}
fmt.Println(u1 == u)  // false
fmt.Println(u1 == u2) // true

结构体嵌套

现在我们已经定义一个 user 结构体了,假设我们再定义两个结构体 adminleader,如下:

type admin struct {
    name    string
    age     int
    isAdmin bool
}

type leader struct {
    name     string
    age      int
    isLeader bool
}

那么问题就来了,有两个字段 nameage 被重复定义了多次。

懒是程序员的必修课。有没有什么办法可以复用这两个字段呢?答案就是结构体嵌套。

使用嵌套方式优化后变成了这样:

type admin struct {
    u       user
    isAdmin bool
}

type leader struct {
    u        user
    isLeader bool
}

代码看起来简洁了很多。

匿名成员

但这样依然不是很完美,每次访问嵌套结构体的成员变量时还是有点麻烦。

// 结构体嵌套
a := admin{
    u:       u,
    isAdmin: true,
}
fmt.Println(a) // {{lisi 18} true}
a.u.name = "wangwu"
fmt.Println(a.u.name)  // wangwu
fmt.Println(a.u.age)   // 18
fmt.Println(a.isAdmin) // true

这个时候就需要匿名成员登场了,不指定名称,只指定类型。

type admin1 struct {
    user
    isAdmin bool
}

通过这种方式可以省略掉中间变量,直接访问我们需要的成员变量。

// 匿名成员
a1 := admin1{
    user:    u,
    isAdmin: true,
}
a1.age = 20
a1.isAdmin = false

fmt.Println(a1)         // {{lisi 20} false}
fmt.Println(a1.name)    // lisi
fmt.Println(a1.age)     // 20
fmt.Println(a1.isAdmin) // false

总结

本文介绍了字典和结构体,两种很常用的数据类型。虽然篇幅不长,但基本操作也都包括,写代码肯定是没有问题的。更底层的原理和更灵活的用法就需要大家自己去探索和发现了。

当然,我也会在写完基础专栏之后,分享一些更深层的文章,欢迎大家关注,交流。

到目前为止,数据类型就都介绍完了。

先是学习了基础数据类型,包括整型,浮点型,复数类型,布尔型和字符串型。然后是复合数据类型,包括数组,切片,字典和结构体。

这些都是 Go 的基础,一定要多多练习,熟练掌握。文中的代码我都已经上传到 Github 了,有需要的同学可以点击文末地址,自行下载。


文章中的脑图和源码都上传到了 GitHub,有需要的同学可自行下载。

地址: https://github.com/yongxinz/gopher/tree/main/sc

关注公众号 AlwaysBeta,回复「goebook」领取 Go 编程经典书籍。

Go 专栏文章列表:

  1. 开发环境搭建以及开发工具 VS Code 配置

  2. 变量和常量的声明与赋值

  3. 基础数据类型:整数、浮点数、复数、布尔值和字符串

  4. 复合数据类型:数组和切片 slice

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

推荐阅读更多精彩内容