go中关于defer函数带返回值和不带返回值的问题

        最近学习go语言,也写过不少defer,例如socket通信当成功建立连接后立马来一个defer mSocket.Close()亦或是当http建立连接后来一个defer resp.Close(),go defer的引入简直不要太爽,开发者将资源释放的动作放在defer里面,就再不需要过于关注资源释放的问题了。

        defer好用是好用,但最近看《go程序设计语言》一书关于defer这一块的介绍时,书中写了一个demo,用defer实现了进入函数的打印以及出函数的打印和函数花费的时间,现把代码贴出来

package main

import (
	"log"
	"time"
)

func main(){
	bigSlowOperation()
}

func bigSlowOperation() {
	defer trace("bigSlowOperation")() // don't forget the extra parentheses
	// ...lots of work...
	time.Sleep(10 * time.Second) // simulate slow operation by sleeping
}

func trace(msg string) func() {
	log.Printf("enter %s", msg)
	start := time.Now()
	return func() { log.Printf("exit %s (%s)", msg, time.Since(start)) }
}

        仔细一看,发现defer后面是一个函数(方法),而且这个函数的返回值也是函数,按照常规的思维方式,defer修饰的代码应该在函数(方法)结束的时候才执行,也就是time.Sleep(10 * time.Second)这句代码执行之后再才轮到defer修饰的trace方法执行,这样trace方法中的log.Printf("enter %s", msg)岂不是在程序执行10s后才执行吗?这如何实现杠进入bigSlowOperation就打印呢?下面我们跑一下

2020/11/14 21:12:21 enter bigSlowOperation
2020/11/14 21:12:31 exit bigSlowOperation (10.0141158s)

Process finished with exit code 0

        从运行结果来看,确实在一进入bigSlowOperation,defer后面的trace就执行了但是trace最后那句return却在bigSlowOperation结束时才执行,但是如果defer不带返回值呢?下面代码实验一下

package main

import "fmt"

func main(){
	defer mdefer()
	fmt.Println("hello world")
	return
}

func mdefer(){
	fmt.Println("enter in mdefer")
}

        下面是结果

hello world
enter in mdefer

Process finished with exit code 0

        可以看出不带return的defer后面的函数确实是在原函数结束的时候才执行。后面我也试过defer后面的函数里面只要有return语句,则只有这个return的语句才会在原函数结束时执行;在此特别记录,一面后续踩坑。此外bigSlowOperation这个方法在代码日志这块也比较有借鉴意义。

你可能感兴趣的:(go)