zoukankan      html  css  js  c++  java
  • Go中的函数一些有趣的功能

    Go中的函数

    函数是Go里面的核心设计,它通过关键字func来申明,他的格式如下
    func funcname(input1 type1, input2 type2) (output1 type1, output2 type2) {
             //这里是处理逻辑代码
             //返回多个值
             return value1, value2
    }

    函数有以下特征:

    • 关键字func用来申明一个函数funcname,匿名函数可以没有funcname。
    • 函数可以有一个或者多个参数,每个参数后面带有类型,通过,分隔
    • 函数可以返回多个值
    • 上面返回值申明了两个变量output1和output2,如果你不想申明也可以,直接就两个类型
    • 如果只有一个返回值且不申明返回值变量,那么你可以省略用以包括返回值的括号
    • 如果没有返回值,那么就直接省略最后的返回信息

    函数作为值、类型

    在Go中函数也是一种变量,我们可以通过type来定义他,他的类型就是所有拥有相同的参数,相同的返回值的一种类型 type type_name func(input1 inputType1 [, input2 inputType2 [, ...]) (result1 resultType1 [, ...])

    函数作为类型到底有什么好处呢?那就是可以把这个类型的函数当做值来传递,请看下面的例子。

    package main
    import "fmt"

    type test_int func(int) bool //申明了一个函数类型

    func isOdd(integer int) bool {
            if integer%2 == 0 {
                   return false
           }
           return true
    }

    func isEven(integer int) bool {
           if integer%2 == 0 {
                  return true
           }
           return false
    }

    //申明的函数类型在这个地方当做了一个参数
    func filter(slice []int, f test_int) []int {
           var result []int
           for _, value := range slice {
                  if f(value) {
                         result = append(result, value)
                  }
           }
           return result
    }

    func main(){
           slice := []int {1, 2, 3, 4, 5, 7}
           fmt.Println("slice = ", slice)
           odd := filter(slice, isOdd) //函数当做值来传递了
           fmt.Println("Odd elements of slice are: ", odd)
           even := filter(slice, isEven)//函数当做值来传递了
           fmt.Println("Even elements of slice are: ", even)
    }

    函数当做值和类型在我们写一些通用接口的时候非常有用,通过上面例子我们看到test_int这个类型是一个函数类型,然后两个filter函数的参数和返回值与test_int类型是一样的,但是我们可以实现很多种的逻辑,这样使得我们的程序变得非常的灵活。

    函数返回值是函数的情况

    技术参考: http://www.cnblogs.com/cool-xing/archive/2012/05/19/2509176.html

    假如你拥有一份吃过寿司的人的清单, 你是否能够根据人名确定他是否在清单上? 这是个很简单的问题, 你只需遍历清单. 嗯, 如果你go的功底很弱, 不知道怎么遍历清单那怎么办? 没关系, 我会给你提供一个刷选器:

    func Screen(patients []string) func(string) bool {
             // 定义匿名函数并返回
             return func(name string) bool {
                      for _, soul := range patients {
                               if soul == name {
                                        return true
                               }
                      }
                      return false
             }
    }

    Screen方法会将刷选的函数返回给调用方, 这样你就可以不用懂怎么去遍历清单了, 你只需调用我返回给你的函数就可以:

    // 吃过寿司的人的清单
    those_who_bought_sushi := []string{"Anand", "JoJo", "Jin", "Mon", "Peter", "Sachin"}
    // 得到刷选器函数
    bought_sushi := Screen(those_who_bought_sushi)
    // 调用刷选器函数就可以知道某人是否在清单上
    fmt.Println(bought_sushi("Anand")) // true
    fmt.Println(bought_sushi("Alex")) // false

    闭包

    地球人都知道:函数只是一段可执行代码,编译后就“固化”了,每个函数在内存中只有一份实例,得到函数的入口点便可以执行函数了。go语言中函数可以作为另一个函数的参数或返回值,可以赋给一个变量。函数可以嵌套定义(使用匿名函数),即在一个函数内部可以定义另一个函数,有了嵌套函数这种结构,便会产生闭包问题。如:

    package main
    import "fmt"

    func ExFunc(n int) func() {
             sum:=n
             return func () { //把匿名函数作为值赋给变量a (Go 不允许函数嵌套。
                                       //然而你可以利用匿名函数实现函数嵌套)
                      fmt.Println(sum+1) //调用本函数外的变量
             } //这里没有()匿名函数不会马上执行
    }

    func main() {
             myFunc:=ExFunc(10)
             myFunc()     // 11
             myAnotherFunc:=ExFunc(20)
             myAnotherFunc()    //21
             myFunc()       //11
             myAnotherFunc()   //21
    }

    这里执行结果:

    11


    21

    
11

    
21

    在这段程序中,匿名函数是函数ExFunc的内嵌函数,并且是ExFunc函数的返回值。我们注意到一个问题:这里的匿名内嵌函数中引用到外层函数中的局部变量sum,Go会这么处理这个问题呢?先让我们来看看这段代码的运行结果。当我们调用分别由不同的参数调用ExFunc函数得到的函数时(myFunc(),myAnotherFunc()),得到的结果是隔离的,也就是说每次调用ExFunc函数后都将生成并保存一个新的局部变量sum。其实这里ExFunc函数返回的就是闭包。

    按照命令式语言的规则,ExFunc函数只是返回了内嵌函数InsFunc的地址,在执行InsFunc函数时将会由于在其作用域内找不到sum变量而出错。而在函数式语言中,当内嵌函数体内引用到体外的变量时,将会把定义时涉及到的引用环境和函数体打包成一个整体(闭包)返回。现在给出引用环境的定义就容易理解了:引用环境是指在程序执行中的某个点所有处于活跃状态的约束(一个变量的名字和其所代表的对象之间的联系)所组成的集合。闭包的使用和正常的函数调用没有区别。

    由于闭包把函数和运行时的引用环境打包成为一个新的整体,所以就解决了函数编程中的嵌套所引发的问题。如上述代码段中,当每次调用ExFunc函数时都将返回一个新的闭包实例,这些实例之间是隔离的,分别包含调用时不同的引用环境现场。不同于函数,闭包在运行时可以有多个实例,不同的引用环境和相同的函数组合可以产生不同的实例。

    闭包函数是把创建时,引用到的外部数据复制了一份,与函数一起组成了一个整体。

    闭包函数出现的条件:
    1.被嵌套的函数引用到非本函数的外部数据,而且这外部数据不是“全局变量”
    2.函数被独立了出来(被父函数返回或赋值给其它函数或变量了)

    回来看闭包的定义:闭包是什么,闭包是由函数及其相关的引用环境组合而成的实体(即:闭包=函数+引用环境)。

    对象是附有行为的数据,而闭包是附有数据的行为

    参考: http://www.cnblogs.com/Jifangliang/archive/2008/08/05/1260602.html

    http://blog.sina.com.cn/s/blog_487109d101018fcx.html

    package main
     
    import "fmt"
     
     
    func ExFunc(n int) func() {
        sum := n
        a := func() {
            sum++        //在这里对外部数据加1
            fmt.Println(sum)
        }
        return a
    }
     
    func main() {
        myFunc := ExFunc(10)
        myFunc()
        myAnotherFunc := ExFunc(20)
        myAnotherFunc()
     
        myFunc()
        myAnotherFunc()     //这里得出的结果是22,由此可以证明两点
                            //1.闭包中对外部数据的修改,外部不可见
                            //2.外部数据的值被保存到新建的静态变量中
                        
    }
     
     

    试验

    看下面几种情况,对比执行结果

    package main
     
    import"fmt"
     
    func main(){
        var j int=5
     
        a:=func()func(){
            var i int=10
            fmt.Printf("\neeee:%d\n",j)
            return func(){
                fmt.Printf("i,j:%d,%d\n",i,j)
            }
        }()
     
        a()
        j*=2
        a()
    }
    执行结果:

    eeee:5
    
i,j:10,5
    
i,j:10,10
    
exit code 0, process exited normally.


    
    

    例子二

    package main
     
    import"fmt"
     
    func main(){
        var j int=5
     
        a:=func()func(){
            var i int=10
            fmt.Printf("\neeee:%d\n",j)
            return func(){
                fmt.Printf("i,j:%d,%d\n",i,j)
            }
        }
     
        a()
        j*=2
        a()
    }
    执行结果:

    eeee:5

    eeee:10
    exit code 0, process exited normally.

    例子三

    package main
     
    import"fmt"
     
    func main(){
        var j int=5
     
        a:=func() func(){
            var i int=10
            fmt.Printf("\neeee:%d\n",j)
            return func(){
                fmt.Printf("i,j:%d,%d\n",i,j)
            }
        }
     
        a()()
        j*=2
        a()()
    }
    执行结果:

    eeee:5
    i,j:10,5

    eeee:10
    i,j:10,10
    exit code 0, process exited normally.

    
    
    
    
    

    参考资料:

    https://github.com/astaxie/build-web-application-with-golang/blob/master/02.3.md

  • 相关阅读:
    Mosaic 前端微服务框架
    使用skipper 扩展fabio 的路由&&http proxy 功能
    Introducing Makisu: Uber’s Fast, Reliable Docker Image Builder for Apache Mesos and Kubernetes
    lua-resty-shell 多任务执行
    openresty 使用lua-resty-shell 执行shell 脚本
    ncm 让跨项目配置一致性简单的工具
    lapis 项目添加prometheus 监控集成grafana
    使用prometheus+ grafana+nginx-module-vts 模块监控openresty
    两天快速开发一个自己的微信小程序
    笔记本如何查看mac地址
  • 原文地址:https://www.cnblogs.com/ghj1976/p/2909364.html
Copyright © 2011-2022 走看看