首页 > 其他分享 >defer有什么用呢

defer有什么用呢

时间:2023-05-14 19:01:44浏览次数:38  
标签:语句 defer 函数 什么 Println 执行 panic

1. 简介

本文将从一个资源回收问题引入,引出defer关键字,并对其进行基本介绍。接着,将详细介绍在资源回收、拦截和处理panic等相关场景下defer的使用。

进一步,介绍defer的执行顺序,以及在注册defer函数时,其参数的求值时机等相关特性。最后,重点讲解defer的注意点,如在defer中函数中需要尽量避免引起panic,以及尽量避免在defer中使用闭包。

通过本文的阅读,读者将对Go语言中的defer有更深入的了解,并且能够更加有效地使用这个关键字。

2. 问题引入

开发过程中,函数可能会打开文件、建立网络连接或者其他需要手动关闭的资源。当函数在处理过程中发生错误时,我们需要手动释放这些资源。而如果有多处需要进行错误处理,手动释放资源将是一个不小的心智负担。同时,如果我们遗漏了资源的释放,就会导致资源泄漏的问题。这种问题可能会导致系统性能下降、程序运行异常或者系统崩溃等。

以下是一个示例代码,其中函数打开了一个文件,读取其中的内容并返回:

func ReadFile(filename string) ([]byte, error) {
    f, err := os.Open(filename)
    if err != nil {
        return nil, err
    }

    var content []byte
    _, err = f.Read(content)
    if err != nil {
        // 出现错误,此时调用Close释放资源
        f.Close()
        return nil, err
    }
    // 正常处理结束,也需要调用Close释放资源
    f.Close()
    return content, nil
}

在上面的代码中,我们使用了 os.Open 函数打开一个文件,并在函数返回之前使用 f.Close() 函数手动关闭文件。同时,在出现错误时,我们也调用了f.Close()方法手动关闭了资源。

但是,我们设想一下,如果函数中不仅仅只打开了一个文件,而是同时打开了文件,网络连接,数据库连接等资源,同时假设函数中需要错误处理的地方有5处,此时在错误处理中,来实现对资源的回收是非常大的心智负担,而且一旦在某个错误处理中,忘记对资源的回收,那就代表着资源的泄漏,将会带来一系列的问题。而且,如果在函数执行过程中发生了panic,此时将不会执行错误处理函数,会直接退出,函数打开的文件可能将不会被关闭。

综上所述,我们这里遇到的问题,在于函数处理过程中,会打开一些资源,在函数退出时需要正确释放资源。而释放资源的方式,如果是在每一个错误处理处来对资源进行释放,此时对于开发人员是一个不小的负担;同时对于函数执行过程中发生panic的情况,也无法正常释放资源。

那有什么方式,能够简洁高效得释放资源,无需在函数的多个错误处理处都执行一次资源的回收;同时也能够处理panic可能导致资源泄漏的问题吗? 其实还真有,Go中的defer关键字便非常适合在该场景中使用,下面我先来了解了解defer

3. defer对问题的解决

3.1 defer基本介绍

Go语言中,我们可以在函数体中使用 defer 关键字,来延迟函数或方法的执行。defer 延迟的函数或方法,会在当前函数执行结束时执行,无论函数是正常返回还是异常返回。也就是说,无论在函数中的哪个位置,只要使用了 defer 延迟执行了某个函数或方法,那么这个函数或方法的执行都会被推迟到当前函数执行结束时再执行。

defer 语句的语法很简单,它只需要在需要延迟执行的语句前加上 defer 关键字即可。defer 语句支持执行函数调用和方法调用,也可以在语句中使用函数参数和方法参数等。下面是一个 defer 语句的示例:

func demo() {
    defer fmt.Println("deferred")
    fmt.Println("hello")
}

在上面的示例中,我们使用了 defer 关键字,延迟了 fmt.Println("deferred") 的执行。当函数执行到 defer 语句时,这个语句并不会立即执行,而是被压入一个栈中,等到函数执行结束时,再按照后进先出的顺序依次执行这些被延迟的语句。在这个示例中,fmt.Println("hello") 会先被执行,然后是被延迟的 fmt.Println("deferred")。因此,输出的结果是:

hello
deferred

3.2 defer对上述问题的解决

通过上述描述,我们了解defer函数能够在函数或方法结束前延迟执行,而且无论函数是正常返回还是发生了panicdefer函数都会被执行。

这个特性非常适合用于资源的释放,例如打开的文件、建立的网络连接、申请的内存等等。我们可以在函数或方法中使用defer来延迟释放这些资源,从而避免因为忘记释放而导致的问题,同时也能够在发生异常时正确地释放资源,让代码更加健壮。下面我们使用defer对上面ReadFile函数进行改进,具体做法是在函数中使用defer关键字,将f.Close()操作延迟到函数结束时执行,代码如下:

func ReadFile(filename string) ([]byte, error) {
    f, err := os.Open(filename)
    if err != nil {
        return nil, err
    }
    // 获取到一个资源,便注册资源释放函数
    defer f.Close()

    var content []byte
    _, err = f.Read(content)
    if err != nil {
        return nil, err
    }

    return content, nil
}

在之前的实现中,无论是正常结束还是出现错误,都需要调用f.Close()释放资源。而现在只需要通过defer关键字注册f.Close()函数即可,这样的代码更简洁,更容易维护,并且不会出现资源泄露的问题。

4.defer其他常见用途

defer语句除了用于在函数中释放资源外,还有其他一些场景的用途,如拦截和处理panic,用于函数结束时打印日志等内容,下面将仔细对其进行说明。

4.1 拦截和处理panic

使用defer语句可以在程序出现panic时,及时进行资源回收和错误处理,避免程序因未处理的panic而直接崩溃。具体来说,可以通过在函数开头使用defer语句注册一个函数来捕获panic。当发生panic时,程序会先执行defer语句注册的函数,再进行panic的传递。

例如下面的代码中,函数中使用了defer来捕获panic,并在发生panic时进行了错误处理和资源回收:

func someFunction() {
    defer func() {
        if r := recover(); r != nil {
            log.Println("Recovered from panic:", r)
            // 进行错误处理或者资源回收
        }
    }()
    // 函数代码
    // 可能会出现panic的代码
}

使用defer语句拦截和处理panic的好处是,在出现panic时,程序不会立即崩溃,而是可以通过defer语句进行错误处理和资源回收,保证程序的正常运行和数据的安全性。同时,这种方式也使得代码更加简洁易读,提高了代码的可维护性和可读性。

4.2 实现函数执行时间的计算

在性能测试和优化过程中,我们通常需要知道某个函数或代码段的执行时间。这个时候可以使用defer记录函数执行开始和结束的时间戳,然后计算两者之差,即可得到函数的执行时间。如下:

func foo() {
    defer func() {
        fmt.Println("foo execution time:", time.Since(start))
    }()
    start := time.Now()
    // 函数执行逻辑
}

在上述代码中,我们使用time.Now()函数获取当前时间戳,并将其存储在start变量中。然后,在函数执行结束时,我们在defer语句中定义一个匿名函数,用来计算函数执行时间并输出。在匿名函数中,我们调用time.Since(start)函数来获取当前时间戳与start变量之间的时间差,并将其输出。这样可以帮助我们快速发现程序中耗时较长的代码段,进而进行优化。

总的来说,defer的场景用途还是比较广泛的,可以在需要在函数执行结束后执行某些操作的场景下使用。

5. defer相关特性

5.1 defer的执行顺序

当函数中有多个defer语句时,它们的执行顺序是后进先出的,也就是说最后一个defer语句会最先执行,倒数第二个defer语句会在最后一个defer语句执行完后执行,以此类推。

例如,下面的代码中有三个defer语句:

func main() {
    defer fmt.Println("Third")
    defer fmt.Println("Second")
    defer fmt.Println("First")
    fmt.Println("Hello, defer!")
}

当函数返回时,它们按照后进先出的顺序执行,所以输出结果是:

Hello, World!
First
Second
Third

5.2 注册defer函数时,其参数的求值时机

在注册defer函数时,如果defer函数传入的参数是变量,那么变量的求值顺序与普通函数调用一样,是在函数参数传递之前进行的。例如,假设有如下代码:

func foo() {
    a := 1
    defer func(x int) {
        fmt.Println("x in defer:", x)
    }(a)
    a = 2
    fmt.Println("a before end of function:", a)
}

在这个例子中,变量a在defer函数中被作为参数传递,defer语句中的匿名函数会捕获a的值,并在函数执行结束时打印该值。foo函数执行的结果如下:

a before end of function:2
x in defer:1

因此,可以看出在defer语句中传入的变量是在注册defer函数时进行求值的,而不是在函数执行结束时。

6. defer注意点

6.1 在defer中尽量避免执行可能引起panic的操作

在使用defer语句时,应当尽量避免在其中引起panic。因为当在defer语句中发生panic时,当前defer函数中后续的语句将无法得到执行,可能无法释放已经申请的资源。此时,程序可能会因为资源泄漏等问题而崩溃或产生其他不可预期的后果。举个例子,假设有如下代码:

func main() {
    defer func() {
       if r := recover(); r != nil {
          fmt.Println("Recovered in defer:", r)
       }
    }()
    fmt.Println("Start")
    defer fmt.Println("First Defer")
    defer func() {
       fmt.Println("Second Defer")
       panic("oops")
       fmt.Println("资源回收")
    }()
    fmt.Println("End")
}

这段代码中,我们在第三个defer语句中引发了panic,这时会触发panic机制,第三个defer后续的代码将不会被执行,最后程序会输出如下结果:

Start
End
Second Defer
First Defer
Recovered in defer: oops

可以看到,第三个defer语句中,由于panic导致了fmt.Println("资源回收")语句无法被执行。因此,在编写代码时,我们应该尽量避免在defer中引起panic,如果不可避免有panic可能性的出现,此时应该对其进行处理,以确保程序的稳定性和可靠性。

6.2 尽量避免在defer中使用闭包

这里先简单介绍下闭包,在 Go 中,闭包是一个函数值(function value),它引用了函数体之外的变量。这个被引用的变量会被“捕获”到闭包中,即使这个变量在闭包被创建之后发生了变化,闭包中也能访问到变化后的值。

defer中使用闭包可能会导致一些意想不到的问题。因为闭包引用了外部变量,而在defer函数执行时,这些变量的值可能已经被修改或者不再存在,从而导致出现不可预期的行为。

举个例子,假设有一个defer函数使用了闭包来记录当前时间戳和某个变量的值:

func foo() {
    i := 0
    defer func() {
        fmt.Printf("i: %d, timestamp: %d\n", i, time.Now().UnixNano())
    }()
    i++
}

在这个例子中,我们使用了闭包来捕获了变量i和当前时间戳,并在defer函数中输出它们的值。然而,由于defer函数的执行时机是在函数返回之后,我们无法确定变量i的值是否已经被修改了。因此,这个例子可能输出的结果是不稳定的,无法得到预期的结果。

因此,尽量避免在defer中使用闭包,可以避免一些潜在的问题。如果必须要使用闭包,那么要格外小心,确保在defer函数执行时闭包引用的变量值仍然是符合预期的。

7. 总结

在本文中,我们从一个资源回收的问题引出了defer,介绍了defer的基本用法以及在资源回收、拦截和处理panic等场景中的使用。我们还讨论了defer的一些特性,如执行顺序以及注册defer函数时,参数的求值时机。最后,我们提醒了在使用defer时需要注意的一些问题,如尽量避免在defer中引起panic和避免在defer中使用闭包。

总的来说,defer是Go语言中一个非常方便和强大的语法特性,在某些场景下可以帮助我们更好地实现某些功能。但是,在使用defer时需要注意一些问题,避免引起不必要的麻烦。掌握defer的使用技巧,可以让我们的代码更加健壮、清晰和易于维护。

标签:语句,defer,函数,什么,Println,执行,panic
From: https://www.cnblogs.com/chenjiazhan/p/17399895.html

相关文章

  • fl studio 需要什么配置,flstudio 21如何设置成中文
    FLStudio21是全功能的音乐工作站,漂亮的大混音盘,先进的制作工具,让你的音乐突破想象力的限制。FLStudio21安装前需要先检查现有的系统配置要求是否符合软件要求。flstudio21需要什么配置FLStudiofor21Windows版:Windows7(SP1+platformupdate),Windows8.1或Windows......
  • 什么是无货源电商
     什么是无货源电商 无货源电商是指没有自己的库存,而是通过第三方供应商来提供商品的电子商务模式。无货源电商的优势在于可以提供更多的商品种类,而且不需要投入大量的资金进行库存管理,可以节省成本。 无货源分两种 第一种店群模式,也就是现在各大平台疯狂收割的一种方式,......
  • 数据库什么时候分库分表
         分库分表实操-=====================容易造成读写都在最新的范围区间内的表,并未起到均分 ------哈希切分 双写,新旧数据库都要同步数据 重点,不会每个写操作都加代码,而是通过aop的方式还需要全量数据迁移 验证新库数据  分库分表工具,使用方无......
  • Ep_操作系统面试题-什么是协程
     协程是一种比线程更加轻量级的存在,一个线程可以拥有多个协程。是一个特殊的函数,这个函数可以在某个地方挂起,并且可以重新在挂起处外继续运行。协程不是被操作系统内核所管理,而完全是由程序所控制(也就是在用户态执行),**不会像线程切换那样消耗资源。**面试宝典 很多人不......
  • jar包是什么?
    JAR文件就是JavaArchiveFile。因为jar包主要是对class文件进行打包,而java编译生成的class文件是平台无关的,这就意味着jar包是跨平台的。JAR文件格式以流行的ZIP文件格式为基础。与ZIP文件不同的是,JAR文件不仅用于压缩和发布,而且还用于部署和封装库、组件和插件程序,并可......
  • 操作系统:为什么一个页表项通常占用1Byte,1字节呢?页表项≠逻辑地址!常见OS内存管理误区辨
    为什么一个页表项通常占用1Byte,1字节呢?页表项≠逻辑地址!如果还不懂,请看:页目录、页表和页三者的关系详解每个页表项占用一个字节是怎么来的?问题启发一开始是做题的时候,为什么不是2^12bit,而是2^12Byte某计算机主存按节址,逻地址和物理地址都是32位页表项大小为4B。请回......
  • 为什么编程语言中,标识符不能以数字开头?
    标识符不能以数字为开头,是为了简化词法解析器设计和实现,规避词法解析中以数字开头的变量与数字解析冲突的问题。如果两种类型的词,如果起始符号不同,那么可以很容易把二者区分开;如果起始符号相同,那么以下符号:234到底是变量还是数字常量?这种情况下,需要根据上下文判断。这就离“......
  • 大公司为什么禁止SpringBoot项目使用Tomcat?
    前言在SpringBoot框架中,我们使用最多的是Tomcat,这是SpringBoot默认的容器技术,而且是内嵌式的Tomcat。同时,SpringBoot也支持Undertow容器,我们可以很方便的用Undertow替换Tomcat,而Undertow的性能和内存使用方面都优于Tomcat,那我们如何使用Undertow技术呢?本文将为大家细细讲解。Spr......
  • 【❂Java集合】循环链表和双向链表的区别是是什么
    最后一个结点指针指向不同在建立一个循环链表时,必须使其最后一个结点的指针指向表头结点,而不是像双向链表那样置为NULL。此种情况还用于在最后一个结点后插入一个新的结点。判断链域值不同在判断是否到表尾时,是判断该结点链域的值是否是表头结点,当链域值等于表头指针时,说明已到......
  • 什么场景下值得使用 const that = this ?
    在Vue2项目中,我们经常会遇到这样一个问题:什么场景下值得用constthat=this?这个问题涉及到JavaScript中的this指针和Vue中的模板语法。本文将从以下三个方面来论述这个问题:this指针的含义和变化Vue模板语法中如何使用thisconstthat=this的作用和优劣首先,我们......