导航

Go中的函数一些有趣的功能

Posted on 2013-02-08 18:20  蝈蝈俊  阅读(4607)  评论(0编辑  收藏  举报

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