What is defer?
Defer is a mechanism provided by the Go language to register delayed calls: functions or statements can be executed after the current function has finished executing (either by a return or by a panic exception).
The defer statement is typically used in pairs: open/close connections; Lock/release lock; Open file/close file, etc.
Defer is useful in scenarios where resources need to be reclaimed, and you can easily do some cleanup before the function ends. On the next line of the open resource statement, simply defer can elegantly close the resource before the function returns.
f, _ := os.Open("defer.txt")
defer f.Close()
Copy the code
Note: in the above code, err is ignored. In fact, it should first check whether there is an error. If there is an error, return directly. If f is null, then you can’t call f.close (), it will panic.
Why do WE need to defer?
When programmers are programming, they often need to open some resources, such as database connections, files, locks, etc. These resources need to be released after use, otherwise they will cause memory leaks.
But programmers are human, and human beings make mistakes. As a result, programmers often forget to close these resources. Golang provides the defer keyword directly at the language level, and on the next line of the open resource statement, you can use the defer statement directly to close the resource after the function is registered. Because of this “little” syntactic candy, programmers forget to write close resource statements much less often.
How can you use defer properly?
The use of defer is pretty simple:
f,err := os.Open(filename)
iferr ! =nil {
panic(err)
}
iff ! =nil {
defer f.Close()
}
Copy the code
Near the statement that opened the file, close it with the defer statement. This way, before the function ends, the statements following defer are automatically executed to close the file.
Of course, there will be a slight delay in defer, which can be avoided by applications with very, very, very high time requirements, and the other delays are generally ignored.
Defer the advanced
What is the underlying rationale behind defer?
Let’s first look at the official explanation of defer:
Each time a “defer” statement executes, the function value and parameters to the call are evaluated as usual and saved anew but the actual function is not invoked. Instead, deferred functions are invoked immediately before the surrounding function returns, in the reverse order they were deferred. If a deferred function value evaluates to nil, execution panics when the function is invoked, not when the “defer” statement is executed.
Each time the defer statement executes, the function is “pushed” and its arguments copied; When the outer function (a non-code block, such as a for loop) exits, the defer function executes in reverse order as defined; If the function deferred executes is nil, panic will occur on the final function call.
Instead of executing immediately, the defer statement goes on a stack and is executed in first in last out order before the return function. That is, the defer statement that was defined first executes last. The reason for this is that functions defined later may depend on resources in front of them and must be executed first. Otherwise, if the preceding function is executed first, the dependency on the following function is gone.
When the defer function definition was introduced, external variables were referred to in two ways, as function parameters and as closures. As a function argument, the value is passed to defer when it is defined and cached; When referenced as a closure, the current value is determined based on the entire context when the function defer is actually called.
The statements that follow defer save the arguments to the function call as they execute, which means they are copied. When executed, the copied variable is actually used, so if the variable is a “value”, it is the same as when it was defined. If this variable is a “reference,” it may not be the same as when it was defined.
Here’s an example:
func main(a) {
var whatever [3]struct{}
for i := range whatever {
defer func(a) {
fmt.Println(i)
}()
}
}
Copy the code
Execution Result:
2
2
2
Copy the code
What defer followed was a closure (as WE’ll see later), where I was a variable of type “reference” and I ended up with a value of 2, so three 2’s were printed.
With that in mind, let’s test the results:
type number int
func (n number) print(a) { fmt.Println(n) }
func (n *number) pprint(a) { fmt.Println(*n) }
func main(a) {
var n number
defer n.print(a)defer n.pprint()
defer func(a) { n.print() ()}defer func(a) { n.pprint() }()
n = 3
}
Copy the code
The execution result is:
3
3
3
0
Copy the code
The fourth defer statement is a closure that references n of the external function, resulting in 3; The third defer statement is the same as the fourth; The second defer statement, where n is a reference, is finally evaluated at 3. The first defer statement, which evaluates directly to n, starts with n=0, so it ends with 0.
Use the defer principle
In some cases, we have deliberately used the evaluation-first, deferred call nature of defer. Imagine a scenario where you open two files to merge, then close the open file handle after the function executes.
func mergeFile(a) error {
f, _ := os.Open("file1.txt")
iff ! =nil {
defer func(f io.Closer) {
iferr := f.Close(); err ! =nil {
fmt.Printf("defer close file1.txt err %v\n", err)
}
}(f)
}
/ /...
f, _ = os.Open("file2.txt")
iff ! =nil {
defer func(f io.Closer) {
iferr := f.Close(); err ! =nil {
fmt.Printf("defer close file2.txt err %v\n", err)
}
}(f)
}
return nil
}
Copy the code
In the code above, we use the principle of defer. When we defined the defer function, the parameters were copied in, and when we actually executed close(), we just closed the correct “file”. Imagine if you didn’t pass in f as a function argument, the last two statements would have closed the same file, the last file to open.
When calling close(), check whether the main body of the call is empty. For example, in the code snippet above, it is safest to call Close() only when f is not empty.
The disassembly of the defer command
If defer were as simple as described above (and it isn’t), the world would be perfect. Things are always not that simple. If we don’t use defer properly, we will jump into a lot of holes.
The key to understanding these pits is this statement:
return xxx
Copy the code
The above statement is compiled into three instructions:
Return value = XXX 2. Call the function defer 3. Empty returnCopy the code
Steps 1,3 are the real commands for the Return statement, and step 2 is the statement defined by defer, where Return values may be manipulated.
Let’s look at two examples of trying to unpack the return statement and defer statement into the correct order.
First example:
func f(a) (r int) {
t := 5
defer func(a) {
t = t + 5} ()return t
}
Copy the code
After the apart:
func f(a) (r int) {
t := 5
// 1. Assign instruction
r = t
// 2. defer is inserted between assignment and return. In this case, the return value r has not been modified
func(a) {
t = t + 5
}
// 3. Empty return instruction
return
}
Copy the code
There is no return value r in the second step, so a call to f() in main yields 5.
Second example:
func f(a) (r int) {
defer func(r int) {
r = r + 5
}(r)
return 1
}
Copy the code
After the apart:
func f(a) (r int) {
/ / 1. Assignment
r = 1
// 2. The changed r is the same as the previous r, and does not change the returned r
func(r int) {
r = r + 5
}(r)
// 3. Empty return
return
}
Copy the code
Therefore, a call to f() in main yields 1.
The parameters of the defer statement
The value of the defer statement expression was determined when it was defined. Here are three functions:
func f1(a) {
var err error
defer fmt.Println(err)
err = errors.New("defer error")
return
}
func f2(a) {
var err error
defer func(a) {
fmt.Println(err)
}()
err = errors.New("defer error")
return
}
func f3(a) {
var err error
defer func(err error) {
fmt.Println(err)
}(err)
err = errors.New("defer error")
return
}
func main(a) {
f1()
f2()
f3()
}
Copy the code
Running results:
<nil>
defer error
<nil>
Copy the code
The first and third functions are evaluated when they are defined as function arguments. When they are defined, the err variables are nil, so they are printed nil. The argument to the second function is also evaluated at definition, but in the second example it is a closure that references the variable err and ends up deferError when executed. Closures are covered later in this article.
The third error is relatively easy to make, and in a production environment, it is easy to write such error code. Finally, the defer statement did not work.
What is a closure?
Closures are entities composed of functions and their associated reference environments, namely:
Closure = function + reference environmentCopy the code
Normal functions have function names, but anonymous functions do not. Anonymous functions cannot stand alone, but can be called directly or assigned to a variable. Anonymous functions are also known as closures, and a closure inherits the scope of the function declaration. In Golang, all anonymous functions are closures.
A less appropriate example is to think of a closure as a class, and a closure function call instantiates a class. Closures can have multiple instances at run time, which capture variables and constants in the same scope and use them wherever the closure is called (instantiated). Also, variables and constants captured by closures are passed by reference, not by value.
Here’s a simple example:
func main(a) {
var a = Accumulator()
fmt.Printf("%d\n", a(1))
fmt.Printf("%d\n", a(10))
fmt.Printf("%d\n", a(100))
fmt.Println("-- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -")
var b = Accumulator()
fmt.Printf("%d\n", b(1))
fmt.Printf("%d\n", b(10))
fmt.Printf("%d\n", b(100))}func Accumulator(a) func(int) int {
var x int
return func(delta int) int {
fmt.Printf("(%+v, %+v) - ", &x, x)
x += delta
return x
}
}
Copy the code
Execution Result:
(0xc420014070, 0) - 1
(0xc420014070, 1) - 11
(0xc420014070, 11) - 111
------------------------
(0xc4200140b8, 0) - 1
(0xc4200140b8, 1) - 11
(0xc4200140b8, 11) - 111
Copy the code
The closure refers to the x variable, and a and B can be considered as two distinct instances that do not affect each other. Inside the instance, the x variable is the same address, so there is an “additive effect.”
Defer cooperate to recover
Golang is often criticized for its errors, which are often all over the place. Programming always returns an error for the caller to handle. If it is the kind of fatal error, such as the implementation of the initialization of the program when the problem, directly panic, save the online run of a bigger problem.
But sometimes we need to recover from exceptions. For example, if the server program encounters a serious problem and panic occurs, we can at least do some “cleaning” before the program crashes, such as closing the client connection and preventing the client from waiting.
Panic stops the currently executing program, not just the current coroutine. Until then, it will execute the statements in the current coroutine defer list in an orderly fashion, with no guarantees for the deferred statements hanging from the other coroutines. As a result, we often hang an Recover statement in defer to prevent the application from hanging directly, and this goes to try… Catch effect.
Note that the recover() function is only valid in the context of defer (and only by using an anonymous function call in defer), and only returns nil if called directly.
func main(a) {
defer fmt.Println("defer main")
var user = os.Getenv("USER_")
go func(a) {
defer func(a) {
fmt.Println("defer caller")
if err := recover(a); err ! =nil {
fmt.Println("recover success. err: ", err)
}
}()
func(a) {
defer func(a) {
fmt.Println("defer here")
}()
if user == "" {
panic("should set user env.")}// Not executed here
fmt.Println("after panic")
}()
}()
time.Sleep(100)
fmt.Println("end of main function")}Copy the code
Panic above will eventually be caught by Recover. This approach is often used in the main flow of an HTTP server. A random request may trigger a bug. In this case, recover is used to catch panic and stabilize the main process without affecting other requests.
The programmer learned the occurrence of the panic through monitoring, located the corresponding position of the log according to the time point, found the cause of the panic, three under five divided by two, repair online. A look around, we are immersed in their own business, it is perfect: secretly fixed a bug, not found! Hey hey!
Afterword.
Defer works very well and generally doesn’t cause problems. But only by understanding how defer works can you avoid its gentle trap. Once you understand how it works, you’ll write easy-to-understand and maintainable code.
The resources
【defer those things 】xiaozhou.net/something-a… Case 】 【 defer code tiancaiamao. Gitbooks. IO/go – internal… [closure] www.kancloud.cn/liupengjie/… [closure] blog.51cto.com/speakingbai… [closure] blog.csdn.net/zhangzhebju… [delay] liyangliang. Me/posts / 2014 /… 【 】 the defer three principles leokongwq. Making. IO / 2016/10/15 /… 【defer code example 】juejin.cn/post/684490… 【defer Panic 】ieevee.com/tech/2017/1… 【 defer panic 】 zhuanlan.zhihu.com/p/33743255