Go语言学习之旅 2 - 基本语法及数据结构
概述
连续三节的内容如下:
- 第一节覆盖了基本语法及数据结构
- 第二节讨论了方法与接口
- 第三节则简单介绍了 Go 的并发原语。
导入包 import
大写开头,导出
如果一个名字以大写字母开头,那么它就是已导出的。例如,Pizza 就是个已导出名,Pi 也同样,它导出自 math 包。
import 分组
import "fmt"
import "math"
不过使用分组导入语句是更好的形式。
import (
"fmt"
"math"
)
函数参数缩写
func add(x int, y int) int {
return x + y
}
可缩写为
func add(x, y int) int {
return x + y
}
有意义的返回值
Go 的返回值可被命名,它们会被视作定义在函数顶部的变量。
var 语句用于声明一个变量
var c, python, java bool
短变量声明
在函数中,简洁赋值语句 := 可在类型明确的地方代替 var 声明。:= 结构不能在函数外使用。
Go 的基本类型有
bool
string
int int8 int16 int32 int64
uint uint8 uint16 uint32 uint64 uintptr
byte // uint8 的别名
rune // int32 的别名
// 表示一个 Unicode 码点
float32 float64
complex64 complex128
类型转换
表达式 T(v) 将值 v 转换为类型 T。
i := 42
f := float64(i)
u := uint(f)
var i int = 42
var f float64 = float64(i)
类型推导
在声明一个变量而不指定其类型时(即使用不带类型的 := 语法或 var = 表达式语法),变量的类型由右值推导得出。
常量
常量的声明与变量类似,只不过是使用 const 关键字。
for 是 Go 中的 “while”
func main() {
sum := 1
for sum < 1000 {
sum += sum
}
fmt.Println(sum)
}
defer
defer 语句会将函数推迟到外层函数返回之后执行。
推迟调用的函数其参数会立即求值,但直到外层函数返回前该函数都不会被调用。
推迟的函数调用会被压入一个栈中。当外层函数返回时,被推迟的函数会按照后进先出的顺序调用。
Go 拥有指针。指针保存了值的内存地址。
类型 *T 是指向 T 类型值的指针。其零值为 nil。
var p *int
& 操作符会生成一个指向其操作数的指针。
i := 42
p = &i
结构体
一个结构体(struct)就是一组字段(field)
数组
数组的长度是其类型的一部分,因此数组不能改变大小
var a [10]int
切片
类型 []T 表示一个元素类型为 T 的切片。
切片通过两个下标来界定,即一个上界和一个下界,二者以冒号分隔:
a[low : high]
a[1:4] 它包含 a 中下标从 1 到 3 的元素:
切片就像数组的引用
切片并不存储任何数据,它只是描述了底层数组中的一段。
更改切片的元素会修改其底层数组中对应的元素。
切片文法类似于没有长度的数组文法。
这是一个数组文法:
[3]bool{true, true, false}
切片下界的默认值为 0,上界则是该切片的长度。
以下切片是等价的:
a[0:10]
a[:10]
a[0:]
a[:]
切片的长度与容量
切片拥有 长度 和 容量。
切片的长度就是它所包含的元素个数。
切片的容量是从它的第一个元素开始数,到其底层数组元素末尾的个数。
切片 s 的长度和容量可通过表达式 len(s) 和 cap(s) 来获取。
nil 切片的零值是 nil
用 make 创建切片,创建动态数组的方式。
make 函数会分配一个元素为零值的数组并返回一个引用了它的切片:
a := make([]int, 5) // len(a)=5
切片的切片,即是二维数组
board := [][]string{
[]string{"_", "_", "_"},
[]string{"_", "_", "_"},
[]string{"_", "_", "_"},
}
Range
for 循环的 range 形式可遍历切片或映射。
当使用 for 循环遍历切片时,每次迭代都会返回两个值。第一个值为当前元素的下标,第二个值为该下标所对应元素的一份副本。
for i, v := range pow {
fmt.Printf("2**%d = %d\n", i, v)
}
可以将下标或值赋予 _ 来忽略它。
for i, _ := range pow
for _, value := range pow
若你只需要索引,忽略第二个变量即可。
for i := range pow
映射 ( map )
其实就是 map 键值对的数据结构实现
映射将键映射到值。
映射的零值为 nil 。nil 映射既没有键,也不能添加键。
make 函数会返回给定类型的映射,并将其初始化备用
var m map[string]Vertex
m = make(map[string]Vertex)
m["Bell Labs"] = xxxxxxx
映射的文法
映射的文法与结构体相似,不过必须有键名。
var m = map[string]Vertex{
"Bell Labs": Vertex{
40.68433, -74.39967,
},
"Google": Vertex{
37.42202, -122.08408,
},
}
若顶级类型只是一个类型名,你可以在文法的元素中省略它。
var m = map[string]Vertex{
"Bell Labs": {40.68433, -74.39967},
"Google": {37.42202, -122.08408},
}
修改映射
在映射 m 中插入或修改元素:
m[key] = elem
获取元素:
elem = m[key]
删除元素:
delete(m, key)
通过双赋值检测某个键是否存在:
elem, ok = m[key]
若 key 在 m 中,ok 为 true ;否则,ok 为 false。
若 key 不在映射中,那么 elem 是该映射元素类型的零值。
函数值
函数也是值。它们可以像其它值一样传递。
函数值可以用作函数的参数或返回值。
PS: 其实就是函数可以当参数传递
函数的闭包
Go 函数可以是一个闭包。闭包是一个函数值,它引用了其函数体之外的变量。该函数可以访问并赋予其引用的变量的值,换句话说,该函数被这些变量“绑定”在一起。
func adder() func(int) int {
sum := 0
return func(x int) int {
sum += x
return sum
}
}
函数的命名返回值
命名返回参数
可以像python那样返回多个结果,只是非tuple
对于不想要的返回值,可以扔垃圾桶_
如果用命名返回参数,return语句可以为空。return 不为空,返回值顺序是return的顺序而非在函数头声明的顺序
package main
func change(a, b int) (x, y int) {
x = a + 100
y = b + 100
return //101, 102
//return x, y //同上
//return y, x //102, 101
}
# 更多学习资料
[http://docscn.studygolang.com/doc/](http://docscn.studygolang.com/doc/)
https://go-zh.org/#
https://tour.go-zh.org/list
END