1 defer语句

defer 语句的用途是:含有 defer 语句的函数或类型方法,会在该函数或方法将要返回之前(return之前),调用另一个函数。所以,**我们可以在函数开始时,将一些容易忘记的结束操作用defer声明好,这样保证这些结束操作能在函数结束后正确执行。**
当defer要调用的函数有参数时,执行 defer 语句的时候,就会对延迟函数的实参进行求值。举例如下

  1. func printA(a int) {
  2. fmt.Println("value of a in deferred function", a)
  3. }
  4. func main() {
  5. a := 5
  6. defer printA(a) //此时a=5,所以实际传到printA中的参数值为5,输出为5
  7. a = 10
  8. fmt.Println("value of a before deferred function call", a)
  9. }

当一个函数内多次调用 defer 时,Go 会把 defer 调用放入到一个栈中,随后按照后进先出(Last In First Out, LIFO)的顺序执行。

2 错误处理

2.1 error类型

在 Go 中,错误一直是很常见的。错误用内建的 error 类型来表示。就像其他的内建类型(如 intfloat64 等),错误值可以存储在变量里、作为函数的返回值等等。error类型定义如下:

type error interface { //是一个接口,实现该接口的类型都可以当作错误类型
    Error() string
}

使用示例:

func main() {  
    f, err := os.Open("/test.txt")
    if err != nil {//error类型
        fmt.Println(err)
        return
    }
    fmt.Println(f.Name(), "opened successfully")//println会自动调用Error()输出错误信息
}

2.2 提取错误信息的方法

Error()返回的描述不是很详细,我们可以用下面的方法获取更详细的错误信息:

  1. 判断底层结构体类型(即实现error接口的结构体),使用结构体字段获取更多信息
  2. 判断底层结构体类型(即实现error接口的结构体),调用类型的方法获取更多信息
  3. 与error类型变量直接比较
    func main() {  
     files, err := filepath.Glob("[")
     if error != nil && err == filepath.ErrBadPattern {//比较,判断是不是某个指定的错误类型
         fmt.Println(err)
         return
     }
     fmt.Println("matched files", files)
    }
    

    2.3 fmt.Errorf

    fmt 包中的 Errorf 函数会根据格式说明符,规定错误的格式,并返回一个符合该错误的字符串。示例: ```go func circleArea(radius float64) (float64, error) {
    if radius < 0 {
     //返回错误类型
     return 0, fmt.Errorf("Area calculation failed, radius %0.2f is less than zero", radius)
    
    } return math.Pi radius radius, nil }

func main() {
radius := -20.0 area, err := circleArea(radius) if err != nil { fmt.Println(err) return } fmt.Printf(“Area of circle %0.2f”, area) }

<a name="BiesT"></a>
# 3 panic和recover
有些情况,当程序发生异常时,无法继续运行。在这种情况下,我们会使用 `panic` 来终止程序。当函数发生 panic 时,它会终止运行,在执行完所有的defer延迟函数后,程序控制返回到该函数的调用方。这样的过程会一直持续下去,直到当前协程的所有函数都返回退出,然后程序会打印出panic信息,接着打印出堆栈跟踪(Stack Trace),最后程序终止。当程序发生 panic 时,使用 `recover` 可以重新获得对该程序的控制。

需要注意:

- **你应该尽可能地使用[错误](https://studygolang.com/articles/12724),而不是使用 panic 和 recover。只有当程序不能继续运行的时候,才应该使用 panic 和 recover 机制**。
- **必须在`defer`函数中直接调用`recover`**。在延迟函数内调用 `recover`,可以取到 `panic` 的错误信息,并且停止panic续发事件(Panicking Sequence),程序运行恢复正常。如果在延迟函数的外部调用 `recover`,就不能停止panic续发事件。
- **只有在相同的Go协程中调用 recover 才管用**。`recover` 不能恢复一个不同协程的 panic



使用示例
```go
func recoverName() {  
    if r := recover(); r!= nil {//恢复程序运行
        fmt.Println("recovered from ", r)
        debug.PrintStack()//打印panic异常的堆栈信息
    }
}

func fullName(firstName *string, lastName *string) {  
    defer recoverName() //延迟函数
    if firstName == nil {
        panic("runtime error: first name cannot be nil")//触发程序异常
    }
    if lastName == nil {
        panic("runtime error: last name cannot be nil")
    }
    fmt.Printf("%s %s\n", *firstName, *lastName)
    fmt.Println("returned normally from fullName")
}

func main() {  
    defer fmt.Println("deferred call in main")
    firstName := "Elon"
    fullName(&firstName, nil)
    fmt.Println("returned normally from main")
}

一个通用的recover defer使用的模板,可供参考:

func foo() (err error) {
    defer func() {
        if r := recover(); r != nil {
            switch x := r.(type) {
            case string:
                err = errors.New(x)
            case error:
                err = x
            default:
                err = fmt.Errorf("Unknown panic: %v", r)
            }
        }
    }()

    panic("TODO")
}