Hello,世界
这是Go语言官方的一个简明入门教程,可以帮助我们快速入门Go语言:Go语言之旅,它提供了一个在线编辑、编译、运行、格式化代码的平台。我们也可以通过
$ go get -u gitHub.com/Go-zh/tour tour
在自己的电脑上安装并运行此教程的程序。
首先让我们来看一下作为各种编程语言入门著名的hello world程序的Go语言版本
package main
import "fmt"
func main() {
fmt.Println("Hello, 世界")
}
上面的代码运行结果输出如下:
Hello, 世界
1 基础
1.1 包
Go语言程序是由包构成的,main()函数作为执行程序的入口,只能包含在main包中。当需要使用外部包的时候,使用import导入外部包。import可能还会包含导入路径,包名与导入路径的最后一个元素一致。当需要导入多个包时,使用import加括号,括号中包含所需要的所有包名。
示例程序:
package main
import (
"fmt"
"math/rand" //导入math/rand包用以调用它定义的Into()函数
)
func main() {
fmt.Println("My favorite number is", rand.Intn(10)) //返回10以内的随机数(没有定义随机数种子,所以每次都返回同一个数-伪随机数)
}
代码运行结果:
My favorite number is 1
1.2 导出名
在Go语言的包中,变量或者函数如果是以大写字母开头的表明它是可以被其他包访问的,如果不是则只能被包内部访问
package main
import (
"fmt"
"math"
)
func main() {
//fmt.Println(math.pi) //这个例子其实并不合适math包里并没有pi这个变量,运行时只会报错undefined。
fmt.Println(math.Pi)
}
1.3 函数
通常把实现某个需要被频繁访问调用的代码封装成一个函数,以使代码变得更简洁高效,Go语言定义函数的方式:
func <func-name> (parameters ) (return values) {
do something
}
下面的代码实现一个加法函数:
package main
import "fmt"
func add(x int, y int) int { //Go语言支持同类型参数,省略类型符号,x int, y int等价于x, y int
return x + y
}
func main() {
fmt.Println(add(42, 13))
}
Go语言函数支持多值返回。
package main
import "fmt"
func swap(x, y string) (string, string) {
return y, x
}
func main() {
a, b := swap("hello", "world") //swap 返回两个值交换后的结果
fmt.Println(a, b)
}
代码运行结果:
world hello
1.4 变量
变量声明格式:
var <var-name> <var-type>
变量初始化:
var <var-name> <var-type> = <var-value>
var <var-name1>, <var-name2> <var-type> = <var-value1>, <var-value2>
短变量声明:Go语言支持不用显式指出变量的类型,它可以根据初始化值的类型做类型自动判断。但是它的使用有一定限制 - 只能在函数体内部使用,如果在函数体外部,必须要使用var关键字。
k := 3 //k会被自动定义为Int类型
当类型推导存在有歧义的可能时,Go语言会根据类型的初始化值的精度判断具体类型。
1.5 基本类型
变量声明也可以同导入语句一样使用分组方式声明(括号),没有明确初始化值的变量会被赋予零值。
- bool - 零值:false
- string - 零值:""
- int(以及各种位数大小不同的整型派生类型)
- byte(8位无符号整型-uint8)
- rune(32位有符号整型-int32)
- float32,float64
- complex64, complex128
数值类型的零值都是0
Go语言中不存在隐式的类型转换,在需要类型转换时必须要显式地使用转换函数(),否则会编译报错:
./prog.go:12:15: cannot use f (variable of type float64) as type uint in variable declaration
1.6 常量
常量使用const关键字定义,且不能使用短变量语法进行声明(很容易理解,不然的话根短变量会产生冲突)
2 流程控制语句
2.1 for循环
Go语言只有for循环,且不需要括号对循环的部分进行包括
普通for循环:
for i := 1; i < 10; i++ {
//do something
}
类C语言while循环:
for i < 100 {
i++
//do something
}
无限循环:
for {
//do something
}
2.2 条件跳转
if语句
if condition {
//do something
}
可以在条件表达式之前执行一个表达式,需要用分号隔开,表达式变量的作用域仅限if之内:
if v := math.Pow(x, n); v < lim {
return v
}
if-else语句
使用else为条件表达式的结果相反时进行另一个操作。
2.3 switch分支跳转
switch相当于一连串的if-else语句,它会从上到下匹配switch表达式的值,匹配成功就执行分支语句块。也可以省略switch表达式,将匹配判断放到case语句中去执行
switch {
case t.Hour() < 12:
fmt.Println("Good morning!")
case t.Hour() < 17:
fmt.Println("Good afternoon.")
default:
fmt.Println("Good evening.")
}
2.4 defer语句
defer语句会将其后面的表达式推迟到外层函数返回之后才运行
package main
import "fmt"
func main() {
defer fmt.Println("world") //这一段会等到main()函数返回之后才执行
fmt.Println("hello")
}
推迟的函数调用会被压入defer栈,根据栈的工作原理,被推迟的函数会按照先进后出的顺序返回。
package main
import "fmt"
func main() {
fmt.Println("counting")
for i := 0; i < 10; i++ {
defer fmt.Println(i)
}
fmt.Println("done")
}
程序运行结果:
counting
done
9
8
7
6
5
4
3
2
1
0
3 指针
指针也是一种变量类型,它存储的是变量或者值的内存地址。
var p *int
指针的零值是nil,&取址操作符,*解引用操作符
i := 42
p = &i //p是一个指针保存了整型变量i的内存地址
*p = 20 //修改了p指向的内存地址出存储的值
与C语言不同的是 Go没有指针运算。(指针自增、自减等)
4 结构体
通过struct关键来定义一组字段创建结构体
type Vertex struct {
x int
y int
}
结构体成员使用.操作符来访问,通过结构体指针来引用结构体成员时可以通过*解引用
package main
import "fmt"
type Vertex struct {
X int
Y int
}
func main() {
v := Vertex{1, 2}
p := &v
(*p).X = 1e9
// p.X = 1e9 为了简化表达,Go语言也支持这样隐式表达,不同于C语言p->X
fmt.Println(v)
}
5 数组
定义数组:
var a [10]int
数组的大小是固定的,不能更改。
6 切片
- 在类似数组的定义时,不限定它的大小,就构成了切片
var s []int
- 切片文法
[]bool{true, false, true}
切片的索引上界是它的长度,下界是0,使用[n:m]表示子切片时,不包括索引值为m的元素。
- 切片长度
len(s)
- 切片容量
cap(s)
- 切片的零值是nil
- 使用make创建切片
a := make ([]int, 5) //创建包含5个整型零值元素的切片,切片的长度是5
a := make([]int, 0, 5) //创建包含0个元素但是容量是5的切片
- 切片本身也可以做为切片的元素,也就是说可以嵌套定义。
- 可以通过内建的append()函数向切片追加元素,如果切片的容量已满,会分配一个更大的数组,返回的切片会指向新分配的数组。但是追加的元素必须要和原切片的类型相同。
7 Range
- range形式的for循环可以用来遍历切片和循环
package main
import "fmt"
var pow = []int{1, 2, 4, 8, 16, 32, 64, 128}
func main() {
for i, v := range pow {
fmt.Printf("2**%d = %d\n", i, v)
}
}
- 可以将索引下标或者值赋予_来忽略它们
for i, _ := range pow
for _, value := range pow
8 映射
- Go语言映射类型就是类似字典那样的键值对,映射的零值是nil,和切片类似,可以使用make()方法初始化映射
type Vertex struct {
Lat, Long float64
}
m := make(map[string]Vertex)
- 访问映射时必须要提供key值
- 映射支持嵌套定义,也就是键值对里的值也可以是映射类型
- 修改映射
m[key] = elem //如果key键存在,则更新对应的值为elem,如果不存在则向映射m添加{key:elem}元素
elem = m[key] //获取映射m中key键对应的值赋给elem
delete(m, key) //删除映射m中key键对应的元素
elem, ok = m[key] //如果key键存在于映射m中,则ok=true,elem被赋予对应的值;否则ok=false,elem赋予对应类型的零值
elem, ok := m[key] //如果elem,ok没有被声明则可以使用短变量声明格式
9 函数值
Go语言中,函数也可以作为一种值类型,它可以用作函数的参数或者返回值
package main
import (
"fmt"
"math"
)
func compute(fn func(float64, float64) float64) float64 {
return fn(3, 4)
}
func main() {
hypot := func(x, y float64) float64 {
return math.Sqrt(x*x + y*y)
} //hypot被定义为一个需要传入两个float64类型参数,返回一个float64类型的函数值类型,它的定义由math.Sqrt()实现
fmt.Println(hypot(5, 12))
fmt.Println(compute(hypot)) //hypot作为函数值类型也可以作为compute函数的参数
fmt.Println(compute(math.Pow))
}
10 函数的闭包
待补充
11 方法
- Go语言没有类的概念,可以为结构体类型定义方法,方法就是带有指定的接收者参数的函数
package main
import (
"fmt"
"math"
)
type Vertex struct { //定义结构体Vertex
X, Y float64
}
func (v Vertex) Abs() float64 { //为结构体Vertex定义方法-方法的接受者必须为Vertex类型变量
return math.Sqrt(v.X*v.X + v.Y*v.Y)
}
func main() {
v := Vertex{3, 4}
fmt.Println(v.Abs())
}
- 并不是只能为结构体类型声明方法,也可以指定接受者类型为内建类型,但是内建类型必须要用type关键字指定别名。话说回来,在Go语言中一旦使用type为内建类型指定别名,这个别名就不能认为是和原内建类型一样的去考虑。
package main
import (
"fmt"
"math"
)
type MyFloat float64 //只有为float64类型指定别名MyFloat之后才能为其定义方法
func (f MyFloat) Abs() float64 {
if f < 0 {
return float64(-f)
}
return float64(f)
}
func main() {
f := MyFloat(-math.Sqrt2)
fmt.Println(f.Abs())
}
- 方法接受者的类型也可以是指针类型,这样方法就可以读取和修改原始值,而不是只在值副本上进行操作
package main
import (
"fmt"
"math"
)
type Vertex struct {
X, Y float64
}
func (v Vertex) Abs() float64 {
return math.Sqrt(v.X*v.X + v.Y*v.Y)
}
func (v *Vertex) Scale(f float64) {
v.X = v.X * f
v.Y = v.Y * f
}
func main() {
v := Vertex{3, 4} //用3,4初始化一个Vertex
v.Scale(10)//因为Scale()方法的接受者是指针类型,所以这里的v是指针,它可以修改原始值3*10,4*10,
fmt.Println(v.Abs())//如果修改Scale()方法使方法接受者为Vertex类型变量,则Scale修改的只是v的副本,在计算v.Abs()时,仍然用的是原始值3,4
}
- 在Go函数中,函数参数如果声明为指针类型必须要传入指针值,否则编译会报错
package main
import (
"fmt"
"math"
)
type Vertex struct {
X, Y float64
}
func Abs(v Vertex) float64 {
return math.Sqrt(v.X*v.X + v.Y*v.Y)
}
func Scale(v *Vertex, f float64) { //如果在这里把*移除,编译就会报错
v.X = v.X * f
v.Y = v.Y * f
}
func main() {
v := Vertex{3, 4}
Scale(&v, 10)
fmt.Println(Abs(v))
}
- 而在方法定义中,即使方法接受者被定义为一个指针,那么在调用方法的类型即使是一个值也不会编译出错
var v Vertex
v.Scale(5) // OK 因为Scale()方法有一个指针接收者,Go会将v.Scale(5)解释为(&v).Scale(5)
p := &v
p.Scale(10) // OK
这样做的原因有二:
首先,方法能够修改接受者指向的值
其次,避免在处理大型结构时,复制值造成的资源浪费