Go语言系列开发之延迟调用和作用域

657 阅读4分钟

Hello,各位小伙伴大家好,我是小栈君,最近一段时间我们将继续分享关于go语言基础系列,当然后期小栈君已经在筹划关于java、Python,数据分析、人工智能和大数据等相关系列文章。希望能和大家一起学习进步,相互提高。

好了,话不多说,我们开始今天的分享,今天分享的主题是关于go语言系列中的延迟调用和作用域相关的知识。

延迟调用defer

在go语言中关键字 defer ⽤于延迟一个函数或者方法(或者当前所创建的匿名函数)的执行。注意,defer语句只能出现在函数或方法的内部。所以我们常用作关闭流处理或则需要进行延期处理的操作。

    func main() {
            defer fmt.Println("小栈君在测试defer") //main结束前调用
            fmt.Println("this is a test")
            /*
                    运行结果:
                    this is a test
                    小栈君在测试defer
            */
    }
 

defer语句经常被用于处理成对的操作,如打开、关闭、连接、断开连接、加锁、释放锁。通过defer机制,不论函数逻辑多复杂,都能保证在任何执行路径下,资源被释放。释放资源的defer应该直接跟在请求资源的语句后。

多个defer执行顺序

如果一个函数中有多个defer语句,它们会以LIFO(后进先出)的顺序执行。哪怕函数或某个延迟调用发生错误,这些调用依旧会被执⾏。

        func test(x int) {
    fmt.Println(10 / x)//x为0时,产生异常
    }
    func main() {
            defer fmt.Println("111")
            defer fmt.Println("222")
            defer test(0)
            defer fmt.Println("333")
            /*
            运行结果:
            333
            222
            111
            panic: runtime error: integer divide by zero
            */
    }

defer和匿名函数结合使用小栈君这边直接为大家举例说明在匿名函数中使用defer 会有怎样的效果呢?

                func main() {
                    a, b := 10, 20
                    defer func(x int) { // a以值传递方式传给x
                            fmt.Println("defer:", x, b) // b 闭包引用
                    }(a)
                    a += 10
                    b += 100
                    fmt.Printf("a = %d, b = %d\n", a, b)
                    /*
                            运行结果:
                            a = 20, b = 120
                            defer: 10 120
                    */
            }

所以在运行结果上看,使用了defer会先进行传值,然后是最后才会计算出来

获取命令行参数

        package main
                import (
                "fmt"
                "os"
                )
                    func main() {
                    // IT小栈君
                    agers := os.Args
                    // 如果用户输入不正确或个数不对 则提醒
                    if agers != nil && len(agers) < 2 {
                    fmt.Println("小栈君提醒你,您输入的位数不正确~")
                    return
                    }
                    fmt.Println("小栈君告诉你实际传参==》", agers)
                    }

运行结果如下:

file

作用域作用域为已声明标识符所表示的常量、类型、变量、函数或包在源代码中的作用范围。就好比是你在行车过程中所能够行使的范围一样。

局部变量

在函数体内声明的变量、参数和返回值变量就是局部变量,它们的作用域只在函数体内:

    func test(a, b int) {
            var c int
            a, b, c = 1, 2, 3
            fmt.Printf("a = %d, b = %d, c = %d\n", a, b, c)
    }
    func main() {
            //a, b, c = 1, 2, 3 //err, a, b, c不属于此作用域
            {
                    var i int
                    i = 10
                    fmt.Printf("i = %d\n", i)
            }
            //i = 20 //err, i不属于此作用域
            if a := 3; a == 3 {
                    fmt.Println("a = ", a)
            }
            //a = 4 //err,a只能if内部使用
    }
 

全局变量

在函数体外声明的变量称之为全局变量,全局变量可以在整个包甚至外部包(被导出后)使用。var a int //全局变量的声明

    func test() {
            fmt.Printf("test a = %d\n", a)
    }
    func main() {
            a = 10
            fmt.Printf("main a = %d\n", a) //main a = 10
            test() //test a = 10
    }

不同作用域同名变量

在不同作用域可以声明同名的变量,其访问原则为:在同一个作用域内,就近原则访问最近的变量,如果此作用域没有此变量声明,则访问全局变量,如果全局变量也没有,则报错。var a int //全局变量的声明

    func test01(a float32) {
            fmt.Printf("a type = %T\n", a) //a type = float32
    }
    func main() {
            fmt.Printf("a type = %T\n", a) //a type = int, 说明使用全局变量的a
            var a uint8 //局部变量声明
            {
        var a float64                  //局部变量声明
        fmt.Printf("a type = %T\n", a) //a type = float64
            }
            fmt.Printf("a type = %T\n", a) //a type = uint8
            test01(3.14)
            test02()
    }
        func test02() {
        fmt.Printf("a type = %T\n", a) //a type = int
    }

今天的分享就到这里就结束啦,如果你喜欢我的分享,麻烦你点击再看,分享或留言,我是小栈君,我们下期见,拜了个拜~

file

本文由博客一文多发平台 OpenWrite 发布!