Go language defer use.

Source: Internet
Author: User
This is a creation in Article, where the information may have evolved or changed.

Defer

There is a good design in the go language, the deferred (defer) statement, where you can add multiple defer statements to a function. When the function executes to the last, these defer statements are executed in reverse order, and finally the function returns. Especially when you are doing some open resources operation, encountered errors need to return early, before returning you need to close the corresponding resources, otherwise it is easy to cause problems such as resource leakage. As shown in the following code, we generally write open a resource that works like this:

Func ReadWrite () bool {    file. Open ("file")//Do some work    if Failurex {        file. Close ()        return False    }    if Failurey {        file. Close ()        return False    }    file. Close ()    return True}

We see a lot of repetitive code on it, Go'sdeferEffectively solves this problem. With it, not only is the amount of code reduced a lot, but the program becomes more elegant. IndeferThe function specified later is called before the function exits.

Func ReadWrite () bool {    file. Open ("file")    defer file. Close ()    if Failurex {        return false    }    if Failurey {        return false    }    return True}

If there are many callsdeferSodeferis in LIFO mode, so the following code will output4 3 2 1 0

For I: = 0; I < 5; i++ {    defer FMT. Printf ("%d", I)}


The first impression that defer gave me was that, like in Java,

try {

}finally {

}

My current understanding is that in the function block using defer, is the function corresponding to a stack space, advanced and out. Call the stack at the end of the function to start the defer operation.

If an object is created that consumes memory and needs to be closed in a timely manner, defer cannot be as accurate as try Finnaly.

[Plain]View Plaincopyprint?
  1. Package Main
  2. Import "FMT"
  3. Import "Time"
  4. Type User struct {
  5. Username string
  6. }
  7. Func (this *user) Close () {
  8. Fmt. Println (this.username, "Closed!!!")
  9. }
  10. Func Main () {
  11. U1: = &user{"Jack"}
  12. Defer U1. Close ()
  13. U2: = &user{"Lily"}
  14. Defer U2. Close ()
  15. Time. Sleep (Ten * time. Second)
  16. Fmt. PRINTLN ("Done!")
  17. }
  18. [Vicky@localhost goroutine]$
Package Mainimport "FMT" Import "time" type User struct {        username string}func (this *user) Close () {        fmt. Println (this.username, "Closed!!!")} Func Main () {        u1: = &user{"Jack"}        defer U1. Close ()        u2: = &user{"Lily"}        defer U2. Close () Time        . Sleep (Ten * time. Second)        FMT. PRINTLN ("Done!")} [Vicky@localhost goroutine]$


[Vicky@localhost goroutine]$ Go Run defertest1.go
Done!
Lily Closed!!!
Jack Closed!!!
[Vicky@localhost goroutine]$


In fact, the thread sleep 10 seconds, U1, and U2 can already close (), but need to rely on the end of the main () function in order to defer execution.

Then try adding an internal code area to defer:


[Plain]View Plaincopyprint?
  1. Package Main
  2. Import "FMT"
  3. Import "Time"
  4. Type User struct {
  5. Username string
  6. }
  7. Func (this *user) Close () {
  8. Fmt. Println (this.username, "Closed!!!")
  9. }
  10. Func Main () {
  11. {
  12. Even with the code fast range, the main function body will still have to end before executing defer
  13. U1: = &user{"Jack"}
  14. Defer U1. Close ()
  15. }
  16. U2: = &user{"Lily"}
  17. Defer U2. Close ()
  18. Time. Sleep (Ten * time. Second)
  19. Fmt. PRINTLN ("Done!")
  20. }
Package Mainimport "FMT" Import "time" type User struct {        username string}func (this *user) Close () {        fmt. Println (this.username, "Closed!!!")} Func Main () {        {                ///Even if the code is fast-range, it still needs the main function body to end before executing defer                u1: = &user{"Jack"}                defer U1. Close ()        }        U2: = &user{"Lily"}        defer U2. Close () Time        . Sleep (Ten * time. Second)        FMT. PRINTLN ("Done!")}

Done!
Lily Closed!!!
Jack Closed!!!
[Vicky@localhost goroutine]$


Still defer's execution in done! After. So how do you get to the exact close of try finally?

[Plain]View Plaincopyprint?
  1. Package Main
  2. Import "FMT"
  3. Import "Time"
  4. Type User struct {
  5. Username string
  6. }
  7. Func (this *user) Close () {
  8. Fmt. Println (this.username, "Closed!!!")
  9. }
  10. Func Main () {
  11. U1: = &user{"Jack"}
  12. F (U1)//In such a way that U1 does not rely on the execution of the main function
  13. In this way, U2 does not rely on the execution of the main function
  14. U2: = &user{"Lily"}
  15. M: = Func () {
  16. Defer U2. Close ()
  17. U2 do somthing
  18. // }
  19. M () <pre class=plain name= "code" > func () {
  20. Defer U2. Close ()
  21. U2 do somthing
  22. } () </PRE> time. Sleep (Ten * time. Second) fmt. PRINTLN ("Done!")} Func f (U *user) {defer u.close ()//U1 do gomething}
Package Mainimport "FMT" Import "time" type User struct {username String}func (this *user) Close () {fmt. Println (this.username, "Closed!!!")}        Func Main () {u1: = &user{"Jack"} f (U1)//In such a way that U1 will not rely on the execution of the main function//Such a way, U2 will not rely on the execution of the main function U2: = &user{"Lily"}//M: = Func () {//defer U2. Close ()/////U2 do somthing//}//M () <div class= "Dp-highlighter bg_plain" ><div cl ass= "Bar" ><div class= "Tools" ><strong>[plain]</strong> <a target=_blank class= "Viewsource" title= "View plain" href= "http://blog.csdn.net/eclipser1987/article/details/12089271#" >view plain</a>< A target=_blank class= "CopyToClipboard" title= "copy" href= "http://blog.csdn.net/eclipser1987/article/details/ 12089271# ">copy</a><a target=_blank class=" Printsource "title=" print "href=" http://blog.csdn.net/ eclipser1987/article/details/12089271# ">print</a><a Target=_blank CLass= "About" title= "?" href= "http://blog.csdn.net/eclipser1987/article/details/12089271#" >?</a><a Target=_blank href= "http://blog.csdn.net/eclipser1987/article/details/12089271" ></a></div></ Div><ol><li class= "alt" ><span><span>func ()  {  </span></span ></li><li><span>         defer u2. Close ()   </span></li><li class= "alt" ><span>          // u2 do somthing  </span></li><li><span } ()   </span></li></ol></div><pre style= "Display:none" class= "plain" name = "Code" > func () {defer U2. Close ()//U2 do somthing} ()
Time. Sleep (Ten * time. Second) fmt. PRINTLN ("Done!")} Func f (U *user) {defer u.close ()//U1 do gomething}

[Vicky@localhost goroutine]$ Go Run defertest3.go
Jack Closed!!!
Lily Closed!!!
Done!


The use of such a method, depending on the unreasonable, but there is the need for existence. In most cases, it can be used in situations such as u1,u2 that consume very much memory, or the CPU, and then execute the time process without too many associations. It retains the functional characteristics of the defer and satisfies the conditions of precise control of the range!

Contact Us

The content source of this page is from Internet, which doesn't represent Alibaba Cloud's opinion; products and services mentioned on that page don't have any relationship with Alibaba Cloud. If the content of the page makes you feel confusing, please write us an email, we will handle the problem within 5 days after receiving your email.

If you find any instances of plagiarism from the community, please send an email to: info-contact@alibabacloud.com and provide relevant evidence. A staff member will contact you within 5 working days.

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.