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语句能非常方便的处理资源释放问题。。比如:资源清理、文件关闭、解锁及记录时间等。

SRE实战 互联网时代守护先锋,助力企业售后服务体系运筹帷幄!一键直达领取阿里云限量特价优惠。

defer执行实际和案例分析

defer执行时机

在Go语言的函数中return语句在底层并不是院子操作,它分为给返回值赋值和RET指令两步。而defer语句执行的实际就在返回值赋值操作后,RET指令执行前。具体如下图所示:
go--defer语句、包 随笔 第1张

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语言为我们提供了很多内置包,如fmtosio等。

定义包

我们还可以根据自己的需要创建自己的包。一个包可以简单理解为一个存放.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()函数在程序运行时自动被调用执行,不能在代码中主动调用它。

包初始化执行的顺序如下图所示:
go--defer语句、包 随笔 第2张

init()函数执行顺序

Go语言包会从main包开始检查其导入的所有包,每个包中又可能导入了其他的包。Go编译器由此构建出一个树状的包引用关系,再根据引用顺序决定编译顺序,依次编译这些包的代码。

在运行时,被最后导入的包会最先初始化并调用其init()函数,如下图所示:
go--defer语句、包 随笔 第3张

扫码关注我们
微信号:SRE实战
拒绝背锅 运筹帷幄