字典与布尔类型
1. 字典
字典(Map 类型),是由若干个 key:value
这样的键值对映射组合在一起的数据结构。
它是哈希表的一个实现,这就要求它的每个映射里的key,都是唯一的,可以使用 ==
和 !=
来进行判等操作,换句话说就是key必须是可哈希的。
什么叫可哈希的?简单来说,一个不可变对象,都可以用一个哈希值来唯一表示,这样的不可变对象,比如字符串类型的对象(可以说除了切片、 字典,函数之外的其他内建类型都算)。
意思就是,你的 key 不能是切片,不能是字典,不能是函数。
字典由key和value组成,它们各自有各自的类型。
在声明字典时,必须指定好你的key和value是什么类型的,然后使用 map 关键字来告诉Go这是一个字典。
map[KEY_TYPE]VALUE_TYPE
声明初始化字典
三种声明并初始化字典的方法
// 第一种方法 var scores map[string]int = map[string]int{"english": 80, "chinese": 85} // 第二种方法 scores := map[string]int{"english": 80, "chinese": 85} // 第三种方法 scores := make(map[string]int) scores["english"] = 80 scores["chinese"] = 85
要注意的是,第一种方法如果拆分成多步(声明、初始化、再赋值),和其他两种有很大的不一样了,相对会比较麻烦(具体请看注释)。
import "fmt" func main() { // 声明一个名为 score 的字典 var scores map[string]int // 未初始化的 score 的零值为nil,无法直接进行赋值 if scores == nil { // 需要使用 make 函数先对其初始化 scores = make(map[string]int) } // 经过初始化后,就可以直接赋值 scores["chinese"] = 90 fmt.Println(scores) }
字典的相关操作
添加元素
scores["math"] = 95
更新元素,若key已存在,则直接更新value
scores["math"] = 100
读取元素,直接使用 [key]
即可 ,如果 key 不存在,也不报错,会返回其value-type 的零值。
fmt.Println(scores["math"])
删除元素,使用 delete 函数,如果 key 不存在,delete 函数会静默处理,不会报错。
delete(scores, "math")
当访问一个不存在的key时,并不会直接报错,而是会返回这个 value 的零值,如果 value的类型是int,就返回0。
package main import "fmt" func main() { scores := make(map[string]int) fmt.Println(scores["english"]) // 输出 0 }
判断 key 是否存在
当key不存在,会返回value-type的零值 ,所以你不能通过返回的结果是否是零值来判断对应的 key 是否存在,因为 key 对应的 value 值可能恰好就是零值。
其实字典的下标读取可以返回两个值,使用第二个返回值都表示对应的 key 是否存在,若存在ok为true,若不存在,则ok为false
import "fmt" func main() { scores := map[string]int{"english": 80, "chinese": 85} math, ok := scores["math"] if ok { fmt.Printf("math 的值是: %d", math) } else { fmt.Println("math 不存在") } }
我们将上面的代码再优化一下
import "fmt" func main() { scores := map[string]int{"english": 80, "chinese": 85} if math, ok := scores["math"]; ok { fmt.Printf("math 的值是: %d", math) } else { fmt.Println("math 不存在") } }
如何对字典进行循环
Go 语言中没有提供类似 Python 的 keys() 和 values() 这样方便的函数,想要获取,你得自己循环。
循环还分三种
- 获取 key 和 value
import "fmt" func main() { scores := map[string]int{"english": 80, "chinese": 85} for subject, score := range scores { fmt.Printf("key: %s, value: %d\n", subject, scores) } }
-
只获取key,这里注意不用占用符。
import "fmt" func main() { scores := map[string]int{"english": 80, "chinese": 85} for subject := range scores { fmt.Printf("key: %s\n", subject) } }
-
只获取 value,用一个占位符替代。
import "fmt" func main() { scores := map[string]int{"english": 80, "chinese": 85} for _, score := range scores { fmt.Printf("value: %d\n", score) } }
2. 布尔类型
关于布尔值,无非就两个值:true 和 false。只是这两个值,在不同的语言里可能不同。
在 Python 中,真值用 True 表示,与 1 相等,假值用 False 表示,与 0 相等
>>> True == 1 True >>> False == 0 True >>>
而在 Go 中,真值用 true 表示,不但不与 1 相等,并且更加严格,不同类型无法进行比较,而假值用 false 表示,同样与 0 无法比较。
Go 中确实不如 Python 那样灵活,bool 与 int 不能直接转换,如果要转换,需要你自己实现函数。
bool 转 int
func bool2int(b bool) int { if b { return 1 } return 0 }
int 转 bool
func int2bool(i int) bool { return i != 0 }
在 Python 中使用 not 对逻辑值取反,而 Go 中使用 !
符号
import "fmt" var male bool = true func main() { fmt.Println( !male == false) // 或者 fmt.Println( male != false) } // output: true
一个 if 判断语句,有可能不只一个判断条件,在 Python 中是使用 and
和 or
来执行逻辑运算
>>> age = 15 >>> gender = "male" >>> >>> gender == "male" and age >18 False
而在 Go 语言中,则使用 &&
表示且
,用 ||
表示或
,并且有短路行为(即左边表达式已经可以确认整个表达式的值,那么右边将不会再被求值。
import "fmt" var age int = 15 var gender string = "male" func main() { // && 两边的表达式都会执行 fmt.Println( age > 18 && gender == "male") // gender == "male" 并不会执行 fmt.Println( age < 18 || gender == "male") } // output: false // output: true