go:匿名函数与闭包
一、匿名函数
定义:没有函数名的函数。
作用:在go语言中目前了解的作用就是用于构成闭包。
*注:由于javascript不存在块级作用域,故匿名函数常用来包含代码以不污染全局命名空间,运行后销毁环境。
----来自知乎回答:http://www.zhihu.com/question/34649602
使用方法及其原理请参考:http://www.cnblogs.com/chenxianbin89/archive/2010/01/28/1658392.html
使用举例
(1)
a := func() { fmt.Println(1) } a() //输出:1
(2)带参数
b := func(arg int) { fmt.Println(arg) } b(2) //输出:2 (func(arg int) { fmt.Println(arg) })(3) //输出:3
(3)带返回值
c := func() int { fmt.Println(4) return 5 } d := c() //打印输出4,并将5赋值给d fmt.Println(d)
二、闭包(closure)
闭包的理解参考:http://www.cnblogs.com/mzwr1982/archive/2012/05/20/2509295.html
闭包的用途参考:http://blog.csdn.net/sunlylorn/article/details/6534610
和 http://www.cnblogs.com/rainman/archive/2009/05/04/1448899.html
简单来说:
因为把返回的函数赋给了一个变量,虽然函数在执行完一瞬间会销毁其执行环境,
但是如果有闭包的话,闭包会保存外部函数的活动对象(变量),所以如果不把对闭包的引用消除掉,
闭包会一直存在内存中,垃圾收集器不会销毁闭包占用的内存。
----来自知乎回答http://www.zhihu.com/question/34649602
使用举例
(1)
//函数A是一个不带参数,返回值是一个匿名函数,且该函数 //带有一个int类型参数,返回值为一个int类型 func A() func(aa int) int { sum := 0 return func(cc int) int { sum += cc fmt.Println("aa=", aa, "bb=", bb, " sum=", sum) return sum } }//编译错误,提示aa未定义
实际上func(aa int) int只是函数A的返回值,在这里给参数取名无任何作用,反而会影响代码阅读,直接用func(int) int 即可。
更正后:
func A() func(int) int { sum := 0 return func(bb int) int { sum += bb fmt.Println("bb=", bb, "\tsum=", sum) return sum } }
调用1:
func main() { a := A()//定义变量a,并将函数A的返回值赋给a b := a(4) fmt.Println(b) } /* ** 输出: ** bb= 4 sum= 4 ** 4 */
调用2:
func main() { a := A() a(0) a(1) a(5) } /* ** 输出: ** bb= 0 sum= 0 ** bb= 1 sum= 1 ** bb= 5 sum= 6 */
以上调用通过闭包实现了sum的累加
调用3:
func main() { a := A() c := A() a(0) a(5) c(10) c(20) } /* ** 输出: ** bb= 0 sum= 0 ** bb= 5 sum= 5 ** bb= 10 sum= 10 ** bb= 20 sum= 30 */
可以看出,上例中调用了两次函数A,构成了两个闭包,这两个闭包维护的变量sum不是同一个变量。
(2)
func B() []func() { b := make([]func(), 3, 3) for i := 0; i < 3; i++ { b[i] = func() { fmt.Println(i) } } return b } func main() { c := B() c[0]() c[1]() c[2]() } /* ** 输出: ** 3 ** 3 ** 3 */
闭包通过引用的方式使用外部函数的变量。
上例中只调用了一次函数B,构成一个闭包,i 在外部函数B中定义,所以闭包维护该变量 i ,c[0]、c[1]、c[2]中的 i 都是闭包中 i 的引用。
因此执行c:=B()后,i 的值已经变为3,故再调用c[0]()时的输出是3而不是0。
可作如下修改:
func B() []func() { b := make([]func(), 3, 3) for i := 0; i < 3; i++ { b[i] = (func(j int) func() { return func() { fmt.Println(j) } })(i) } return b } func main() { c := B() c[0]() c[1]() c[2]() } /* ** 输出: ** 0 ** 1 ** 2 */
以上修改可能没有什么实际意义,此处仅为说明问题使用。
在使用defer的时候可能出现类似问题,需要注意:
for j := 0; j < 2; j++ { defer (func() { fmt.Println(j) })() } /* ** 输出: ** 2 ** 2 */