在线时间:8:00-16:00
迪恩网络APP
随时随地掌握行业动态
扫描二维码
关注迪恩网络微信公众号
函数是Go里面的核心设计,它通过关键字func来申明,他的格式如下 函数有以下特征:
函数作为值、类型在Go中函数也是一种变量,我们可以通过type来定义他,他的类型就是所有拥有相同的参数,相同的返回值的一种类型 type type_name func(input1 inputType1 [, input2 inputType2 [, ...]) (result1 resultType1 [, ...]) 函数作为类型到底有什么好处呢?那就是可以把这个类型的函数当做值来传递,请看下面的例子。 package main type test_int func(int) bool //申明了一个函数类型 func isOdd(integer int) bool { func isEven(integer int) bool { //申明的函数类型在这个地方当做了一个参数 func main(){ 函数当做值和类型在我们写一些通用接口的时候非常有用,通过上面例子我们看到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 { Screen方法会将刷选的函数返回给调用方, 这样你就可以不用懂怎么去遍历清单了, 你只需调用我返回给你的函数就可以: // 吃过寿司的人的清单 闭包地球人都知道:函数只是一段可执行代码,编译后就“固化”了,每个函数在内存中只有一份实例,得到函数的入口点便可以执行函数了。go语言中函数可以作为另一个函数的参数或返回值,可以赋给一个变量。函数可以嵌套定义(使用匿名函数),即在一个函数内部可以定义另一个函数,有了嵌套函数这种结构,便会产生闭包问题。如: package main func ExFunc(n int) func() { func main() { 这里执行结果: 11 21 11 21 在这段程序中,匿名函数是函数ExFunc的内嵌函数,并且是ExFunc函数的返回值。我们注意到一个问题:这里的匿名内嵌函数中引用到外层函数中的局部变量sum,Go会这么处理这个问题呢?先让我们来看看这段代码的运行结果。当我们调用分别由不同的参数调用ExFunc函数得到的函数时(myFunc(),myAnotherFunc()),得到的结果是隔离的,也就是说每次调用ExFunc函数后都将生成并保存一个新的局部变量sum。其实这里ExFunc函数返回的就是闭包。
按照命令式语言的规则,ExFunc函数只是返回了内嵌函数InsFunc的地址,在执行InsFunc函数时将会由于在其作用域内找不到sum变量而出错。而在函数式语言中,当内嵌函数体内引用到体外的变量时,将会把定义时涉及到的引用环境和函数体打包成一个整体(闭包)返回。现在给出引用环境的定义就容易理解了:引用环境是指在程序执行中的某个点所有处于活跃状态的约束(一个变量的名字和其所代表的对象之间的联系)所组成的集合。闭包的使用和正常的函数调用没有区别。
由于闭包把函数和运行时的引用环境打包成为一个新的整体,所以就解决了函数编程中的嵌套所引发的问题。如上述代码段中,当每次调用ExFunc函数时都将返回一个新的闭包实例,这些实例之间是隔离的,分别包含调用时不同的引用环境现场。不同于函数,闭包在运行时可以有多个实例,不同的引用环境和相同的函数组合可以产生不同的实例。 闭包函数是把创建时,引用到的外部数据复制了一份,与函数一起组成了一个整体。 闭包函数出现的条件: 回来看闭包的定义:闭包是什么,闭包是由函数及其相关的引用环境组合而成的实体(即:闭包=函数+引用环境)。 对象是附有行为的数据,而闭包是附有数据的行为 参考: 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 例子二 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 例子三 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
参考资料: https://github.com/astaxie/build-web-application-with-golang/blob/master/02.3.md
|
请发表评论