52.笔记go语言——defer使用场景

52.笔记go语言——defer使用场景

1. 简化资源的回收

 比如:

mu.Lock()

defer mu.Unlock()

当然, defer 也有一定的开销, 也有为了节省性能而回避使用的 defer 的.

从简化资源的释放角度看, defer 类似一个语法糖, 好像不是必须的.

2. panic异常的捕获

defer 除了用于简化资源的释放外,还是Go语言异常框架的一个组成部分.

Go语言中, panic用于抛出异常, recover用于捕获异常. recover只能在defer语句中使用, 直接调用recover是无效的.

比如:

packagemain
 
import"fmt"
 
funcmain(){
    f()
    fmt.Println("Returnednormallyfromf.")
}
 
funcf(){
    deferfunc(){
        ifr:=recover();r!=nil{
            fmt.Println("Recoveredinf",r)
        }
    }()
    fmt.Println("Callingg.")
    g()
    fmt.Println("Returnednormallyfromg.")
}
 
funcg(){
    panic("ERROR")
}

因此, 如果要捕获Go语言中函数的异常, 就离不开defer语句了.

3.修改返回值

defer 除了用于配合 recover,用于捕获 panic 异常外, 还可以用于在 return 之后修改函数的返回值.

例如:

func doubleSum(a, b int) (sum int) {

    deferfunc() {

        sum *=2

    }()

    sum = a + b

}

4. 安全的回收资源

defer 最常见的用法是简化资源的回收. 而且, 从资源回收角度看, defer 只是一个语法糖.

比如, 有一个线程安全的slice修改函数, 为了性能没有使用defer语句:

func set(mu *sync.Mutex, arr []int, i, v int) {

    mu.Lock()

    arr[i] = v

    mu.Unlock()

}

但是, 如果 i >= len(arr)的话, runtime就会抛出切片越界的异常(这里只是举例, 实际开发中不应该出现切片越界异常). 这样的话, mu.Unlock() 就没有机会被执行了.

如果用defer的话, 即使出现异常也能保证mu.Unlock()被调用:

func set(mu *sync.Mutex, arr []int, i, v int) {

    mu.Lock()

    defermu.Unlock()

    arr[i] = v

}

当然, Go语言约定异常不会跨越package边界. 因此, 调用一般函数的时候不用担心goroutine异常退出的情况.

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值