其实匿名函数就是闭包
写一个不使用闭包的实现的计数器
func add1() {val := 0for i := 0; i < 10; i++ {fmt.Printf("i=%d \t", i)val += add2(i)fmt.Println(val)}fmt.Println("总和为:", val)
}func add2(x int) int {sum := 0sum += xreturn sum
}
当运行add1()函数的时候,会给出结果
i=0 0
i=1 1
i=2 3
i=3 6
i=4 10
i=5 15
i=6 21
i=7 28
i=8 36
i=9 45
总和为: 45
记录了计数并叠加的结果。
对比一下,再写一个使用闭包的计数器,并和上面的输出结果一致
func adder1() {result := adder2()for i := 0; i < 10; i++ {fmt.Printf("i=%d \t", i)fmt.Println(result(i))}
}func adder2() func(int) int {sum := 0result := func(num int) int {sum += numreturn sum}return result
}
最后来写另外一个闭包计数器,当每次调用方法时,计数器加1。
func counter1() {res := counter2()fmt.Printf("%T \n", res)fmt.Println("res:", res) fmt.Println("res:", res()) fmt.Println("res:", res()) fmt.Println("res:", res())
}func counter2() func() int {i := 0res := func() int {i++return i}return res
}
当执行counter1()函数里的res()一次,计数器加1一次,所以输出的结果第一行就是res的数据类型,第二行就是res的内存地址,第三行为1,第四行为2,第五行为3……以此类推。
func() int
res: 0x490070
res: 1
res: 2
res: 3
闭包方式引用defer语句,会导致一些问题:
func testCounter() {for i := 0; i < 3; i++ {defer func() { println(i) }()}
}
运行后,返回如下:
3
3
3
因为是闭包,在for迭代语句中,每个defer语句延迟执行的函数时引用的同一个i迭代变量,所以都是循环结束后的3。
解决方案:
第一,在循环体中,创建一个实体变量,每次在defer语句中打印那个实体变量。
第二,将迭代变量i,通过闭包参数的方式传递进去,defer语句会马上对参数进行求值。
func testCounter01() {for i := 0; i < 3; i++ {//定义一个循环体内局部变量ii := idefer func() { println(i) }()}
}func testCounter02() {for i := 0; i < 3; i++ {//通过函数传入i,defer会马上对调用参数求值defer func(i int) {println(i)}(i)}
}
这两个方法的运行结果都是:
2
1
0
注:这两种方式都是可以工作的,但不建议在for循环中使用defer语句。
2019年9月12日,添加一个闭包新的使用例子:
func testFunction01(suffix string) func(string) string {return func(str string) string {if !strings.HasSuffix(str, suffix) {return str + suffix} else {return str}}
}func main() {tf01 := testFunction01(".aaa")fmt.Println(tf01("aaaaa")) //aaaaa.aaafmt.Println(tf01("bbbbb.aaa")) //bbbbb.aaafmt.Println(tf01("ccccc.ccc")) //ccccc.ccc.aaa
}
该例的作用是判断一串字符串是否以指定的字符串结尾,如果不是则加上指定的字符串,如果是则不变。
(该方法可以判断文件的类型,并修改文件名)
当tf01初始化时,利用闭包驻留的特性,后缀名.aaa就已进入参,再次调用时则不必再次输入。
可反复使用该变量。