Golang Defer关键字特定操作详解

Go语言中的defer关键字用于在函数返回前执行一些特定的操作。可以将defer看作是一种后置语句,在函数中的任何位置都可以使用。

下面是一个使用defer的例子:

func foo() {
    defer fmt.Println("Done")
    fmt.Println("Hello")
}

在上面的例子中,当函数foo被调用时,它会先输出"Hello",然后再输出"Done",因为"Done"被包装在defer语句中,会在函数返回前执行。

如果在一个函数中有多个defer语句,它们的执行顺序是后进先出的,也就是说,最后一个defer语句会最先执行,而第一个defer语句会最后执行。

例如,下面的代码中有三个defer语句,它们的执行顺序是3、2、1:

func bar() {
    defer fmt.Println("1")
    defer fmt.Println("2")
    defer fmt.Println("3")
}

在Go语言中,defer关键字可以用于以下几个应用场景:

资源管理:使用defer语句可以确保资源(如文件、网络连接等)被及时关闭和释放,以避免资源泄漏和占用过多的系统资源。

延迟函数调用:通过defer语句可以将函数调用延迟到当前函数返回之前执行,这在一些需要在函数执行结束前执行一些清理或者统计操作的场景下很有用。

错误处理:通过defer语句可以捕获和处理一些可能发生的错误,确保程序在出现异常情况时也能够正常退出并清理资源。

统计和调试:通过defer语句可以在函数执行过程中记录一些统计信息或者调试信息,以便在需要时进行分析和排查。

应用实例1 错误异常的处理

func readFile(filename string) (string, error) {
    file, err := os.Open(filename)
    if err != nil {
        return "", err
    }
    defer func() {
        if err := file.Close(); err != nil {
            log.Println("Error closing file:", err)
        }
    }()
    content, err := ioutil.ReadAll(file)
    if err != nil {
        return "", err
    }
    return string(content), nil
}

在上面的例子中,我们在打开文件后使用了defer语句将文件关闭的代码放在了一个匿名函数中。如果文件读取操作发生错误,这个匿名函数会在函数返回前被执行,确保文件被关闭并且任何可能的资源泄漏被避免。

需要注意的是,如果在defer语句中使用了函数参数,需要在defer语句执行的时候将它们赋值为函数返回值。在上面的例子中,我们需要将file.Close()的返回值赋给err变量,以便检查文件是否成功关闭。

应用实例2 代码统计和调试中的使用

func processRequest(req *http.Request) (*http.Response, error) {
    startTime := time.Now()
    defer func() {
        log.Printf("Request processed in %v", time.Since(startTime))
    }()
    // Process the request here
    // ...
    resp, err := http.DefaultClient.Do(req)
    if err != nil {
        return nil, err
    }
    return resp, nil
}

在上面的示例代码中,我们使用defer语句记录了函数执行的开始时间,然后在函数返回前打印出函数执行的耗时。这样可以方便地进行性能分析和调试,以便优化代码并找出潜在的性能瓶颈。当函数返回时,defer语句会被执行,打印出函数执行的耗时。由于defer语句的特性,无论函数执行过程中是否发生了异常,这个defer语句都会被执行,确保我们能够正确地记录函数的执行时间。

应用实例3 HTTP请求中的异常处理

func handleRequest(w http.ResponseWriter, r *http.Request) {
    // 将处理异常的代码放在defer语句中
    defer func() {
        if r := recover(); r != nil {
            log.Printf("Recovered from panic: %v", r)
            http.Error(w, "Internal Server Error", http.StatusInternalServerError)
        }
    }()
    // 处理HTTP请求
    // ...
    // 在代码中可能会发生异常
    // ...
}

在上面的示例代码中,我们在处理HTTP请求的代码中可能会发生异常,为了确保在异常发生时仍能够释放资源和清理状态,我们使用defer语句将异常处理的代码放在了一个匿名函数中。当发生异常时,defer语句会被执行,捕获异常并打印出异常信息,并向客户端返回一个500错误响应。

需要注意的是,如果在defer语句中使用了函数参数,需要在defer语句执行的时候将它们赋值为函数返回值。在上面的示例代码中,我们需要将recover()的返回值赋给r变量,以便检查是否发生了异常。同时,由于Go语言中的异常处理机制使用较少,因此在处理HTTP请求时应该特别注意异常情况的处理。

注意

需要注意的是,defer语句应该被谨慎使用,特别是在性能敏感的场景下。因为defer语句会增加代码的执行时间和内存消耗,特别是在循环和递归等高频执行的代码中,过多的defer语句可能会影响代码的性能和稳定性。

到此这篇关于Golang Defer关键字特定操作详解的文章就介绍到这了,更多相关Golang Defer关键字内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Golang的关键字defer的使用方法

    目录 核心思想 defer链 源码分析 优化 核心思想 在defer出现的地方插入了指令CALL runtime.deferproc,在函数返回的地方插入了CALL runtime.deferreturn.goroutine的控制结构中,有一张表记录defer,调用runtime.deferproc时会将需要defer的表达式记录在表中,而在调用runtime.deferreturn的时候,则会依次从defer表中“出栈”并执行 如果有多个defer,调用顺序类似栈,越后面的defer表达式越先

  • Golang Defer关键字特定操作详解

    Go语言中的defer关键字用于在函数返回前执行一些特定的操作.可以将defer看作是一种后置语句,在函数中的任何位置都可以使用. 下面是一个使用defer的例子: func foo() { defer fmt.Println("Done") fmt.Println("Hello") } 在上面的例子中,当函数foo被调用时,它会先输出"Hello",然后再输出"Done",因为"Done"被包装在defe

  • GoLang channel关闭状态相关操作详解

    关于 channel 的使用,有几点不方便的地方: 1.在不改变 channel 自身状态的情况下,无法获知一个 channel 是否关闭. 2.关闭一个 closed channel 会导致 panic.所以,如果关闭 channel 的一方在不知道 channel 是否处于关闭状态时就去贸然关闭 channel 是很危险的事情. 3.向一个 closed channel 发送数据会导致 panic.所以,如果向 channel 发送数据的一方不知道 channel 是否处于关闭状态时就去贸然

  • 深入Golang中的sync.Pool详解

    我们通常用golang来构建高并发场景下的应用,但是由于golang内建的GC机制会影响应用的性能,为了减少GC,golang提供了对象重用的机制,也就是sync.Pool对象池. sync.Pool是可伸缩的,并发安全的.其大小仅受限于内存的大小,可以被看作是一个存放可重用对象的值的容器. 设计的目的是存放已经分配的但是暂时不用的对象,在需要用到的时候直接从pool中取. 任何存放区其中的值可以在任何时候被删除而不通知,在高负载下可以动态的扩容,在不活跃时对象池会收缩. sync.Pool首先

  • Golang之sync.Pool使用详解

    前言 我们通常用 Golang 来开发并构建高并发场景下的服务,但是由于 Golang 内建的GC机制多少会影响服务的性能,因此,为了减少频繁GC,Golang提供了对象重用的机制,也就是使用sync.Pool构建对象池. sync.Pool介绍 首先sync.Pool是可伸缩的临时对象池,也是并发安全的.其可伸缩的大小会受限于内存的大小,可以理解为是一个存放可重用对象的容器.sync.Pool设计的目的就是用于存放已经分配的但是暂时又不用的对象,而且在需要用到的时候,可以直接从该pool中取.

  • Golang信号量设计实现示例详解

    目录 开篇 信号量 semaphore 扩展库实现 Acquire Release TryAcquire 总结 开篇 在我们此前的文章 Golang Mutex 原理解析 中曾提到过,Mutex 的底层结构包含了两个字段,state 和 sema: type Mutex struct { state int32 sema uint32 } state 代表互斥锁的状态,比如是否被锁定: sema 表示信号量,协程阻塞会等待该信号量,解锁的协程释放信号量从而唤醒等待信号量的协程. 这个 sema

  • MySQL中数据视图操作详解

    目录 1.视图概述 1.1创建视图 1.2视图的查询 2.操作视图 2.1通过视图操作数据 2.2修改视图定义 2.3删除视图 1.视图概述 视图是从一个或多个表(或视图)导出的表.视图与表(有时为与视图区别,也称表为基本表)不同,视图是一个虚表,即视图所对应的数据不进行实际存储,数据库中只存储视图的定义,对视图的数据进行操作时,系统根据视图的定义去操作与视图相关联的基本表. 视图一经定义,就可以像表一样被查询.修改.删除和更新.使用视图有下列优点: 1.为用户集中数据,简化用户的数据查询和处理

  • MySQL数据库中表的操作详解

    目录 1.Mysql中的数据类型 2.创建数据表 3.删除表 4.插入数据 5.更新数据 6.删除数据 7.快速复制表 8.快速删除表数据 1.Mysql中的数据类型 varchar 动态字符串类型(最长255位),可以根据实际长度来动态分配空间,例如:varchar(100) char 定长字符串(最长255位),存储空间是固定的,例如:char(10) int 整数型(最长11位) long 长整型 float 单精度 double 双精度 date 短日期,只包括年月日 datetime

  • GoLang内存泄漏原因排查详解

    目录 背景 临时性内存泄漏 通道理解 背景 Go 语言中有对应的Go 内存回收机制,在Go采用 并发三色标记清除  算法, 但是由于实际的过程中 发现会有一些内存泄漏的常见,内存泄漏 分为: 临时性 和 永久性内存泄漏. 初步排查过程中: 发现Linux使用top 发现内存随着时间会持续的增加没有稳定在一个合理值中. 在使用 pprof ,BBC 等 Go的内存泄漏工具进行排查 临时性内存泄漏 指的释放内存 不及时,对应的内存在更晚时候释放,这类问题主要是 string,slice 和底层的Bu

  • Jquery基础之事件操作详解

    事件是用户操作时页面或页面加载时引发的用来完成javascript和HTML之间的交互操作.常见的元素点击事件.鼠标事件.键盘输入事件等,较传Javascript 相比JQuery增加并扩展了基本的事件处理机制,极大的增强了事件处理的能力. 一.DOM加载事件 页面加载完毕后浏览器会通过javascript为Dom元素加载事件,使用Javascript时候使用的是window.onload方法,而Jquery使用的是$(document).ready()方法,下表 展示两个事件的异同. wind

  • 对Golang import 导入包语法详解

    package 的导入语法 写 Go 代码的时经常用到 import 这个命令用来导入包,参考如下: import( "fmt" ) 然后在代码里面可以通过如下的方式调用: fmt.Println( "我爱北京天安门" ) fmt 是 Go 的标准库,它其实是去 GOROOT 下去加载该模块,当然 Go 的 import 还支持如下两种方式来加载自己写的模块: 相对路径 import "./model" // 当前文件同一目录的 model 目录

随机推荐