go--defer语句、包
defer语句
Go语言中的defer
语句会将其后面跟随的语句进行延迟处理。在defer
归属的函数即将返回时,将延迟处理的语句按defer
定义的逆序进行执行,也就是说,先被defer
的语句最后执行,最后被defer
的语句,最先被执行。
func main() {
fmt.Println("start")
defer fmt.Print(1)
defer fmt.Print(2)
defer fmt.Print(3)
fmt.Println("end")
}
// 结果:start end 3 2 1
由于defer
语句延迟调用的特性,所以defer
语句能非常方便的处理资源释放问题。。比如:资源清理、文件关闭、解锁及记录时间等。
defer执行实际和案例分析
defer执行时机
在Go语言的函数中return
语句在底层并不是院子操作,它分为给返回值赋值和RET指令两步。而defer
语句执行的实际就在返回值赋值操作后,RET指令执行前。具体如下图所示:
defer经典案例
func f1() int {
x := 5
defer func() {
x++
}()
return x
}
func f2() (x int) {
defer func() {
x++
}()
return 5
}
func f3() (y int) {
x := 5
defer func() {
x++
}()
return x
}
func f4() (x int) {
defer func(x int) {
x++
}(x)
return 5
}
func main() {
fmt.Println(f1()) // 5
fmt.Println(f2()) // 6(x等于5,返回值等于x,所以x++后返回值会改变)
fmt.Println(f3()) // 5
fmt.Println(f4()) // 5
}
Go语言的包
包的定义
包(package)
是多个Go源码的集合,是一种高级的代码复用方案,Go语言为我们提供了很多内置包,如fmt
、os
,io
等。
定义包
我们还可以根据自己的需要创建自己的包。一个包可以简单理解为一个存放.go
文件的文件夹。该文件夹下面的所有go文件都要在代码的第一行添加如下代码,声明该文件归属的包。
package 包名
注意事项
- 一个文件夹下面只能有一个包,同样一个包的文件不能在多个文件夹下。
- 包名可以不和文件夹的名字一样,包名不能包含
-
符号。 - 包名为main的包为应用程序的入口包,编译不包含main包的源代码时不会得到可执行文件。
可见性
如果想在一个包中引用另外一个包里的标识符(如变量、常量、类型、函数等)时,该标识符必须是对外可见的。在Go语言中只需要将标识符的首字母大写就可以让标识符对外可见了。
package main
import "fmt"
var a = 100 // 首字母小写,外部包不可见,只能在当前包内使用
// 首字母大写外部包可见,可在其他包中使用
const Mode = 1
type person struct { // 首字母小写,外部包不可见,只能在当前包内使用
name string
}
包的导入
要在代码中引用其他包的内容,需要使用import
关键字导入使用的包。具体语法如下:
import "包的路径"
** 注意事项
- import导入语句通常放在文件开头包声明语句的下面。
- 导入的包名需要使用双引号包裹起来。
- 包名是从
$GOPATH/src/
后开始计算的,使用/
进行路径分隔。 - Go语言中禁止循环导入包。
单行导入
单行导入的格式如下:
import "包1"
import "包2"
多行导入
import (
"包1"
"包2"
)
自定义包名
在导入包名的时候,我们还可以为导入的包设置别名。具体语法格式如下:
import 别名 "包的路径"
单行导入方式定义别名:
import "fmt"
import m "github.com/aaa/bbb/ccc"
多行导入方式定义别名:
import (
"fmt"
m "github.com/aaa/bbb/ccc"
)
匿名导入包
如果只希望导入包,而不使用包内部的数据时,可以使用匿名导入包。具体的格式如下:
import _ "包的路径"
init()初始化函数
init()函数介绍
在Go语言程序执行时导入包语句会自动触发包内部init()
函数的调用。需要注意的是:init()
函数没有参数也没有返回值。init()
函数在程序运行时自动被调用执行,不能在代码中主动调用它。
init()函数执行顺序
Go语言包会从main
包开始检查其导入的所有包,每个包中又可能导入了其他的包。Go编译器由此构建出一个树状的包引用关系,再根据引用顺序决定编译顺序,依次编译这些包的代码。
在运行时,被最后导入的包会最先初始化并调用其init()
函数,如下图所示:
