一.闭包
基本介绍
闭包就是一个函数和其他相关的引用环境组合的一个整体(实体)
//案例
package main
import "fmt"
//累加器
func AddUpper() func (int) int {
var n int = 10
return func(x int) int {
n = n + x
return n
}
}
func main() {
//使用前面的代码
f := AddUpper()
fmt.Println(f(1)) //11
fmt.Println(f(2)) //12
fmt.Println(f(3)) //13
}
对上面代码说明:
1.AddUpper是一个函数,返回的数据类型是func (int) int
2.闭包说明
var n int = 10
return func(x int) int {
n = n + x
return n
}
返回的是一个匿名函数,但是这个匿名函数引用到函数外面的n,因此这个匿名函数就和n形成了一个整体,构成闭包
3.理解:闭包是类,函数是操作,n是字段.函数和它使用到的n构成闭包
4.当反复调用f函数时,因为n是初始化一次,因此每调用一次就进行累计
5.要搞清楚闭包的关键,就要分析出返回的函数它使用(引用)到哪些变量,因为函数和它引用到的变量共同构成闭包
//案例
//1.编写一个函数makeSuffix(suffix string) 可以接收一个文件后缀名(eg: .jpg),并返回一个闭包
//2.调用闭包,可以传入一个文件名,如果该文件名没有指定的后缀(eg:.jpg),则返回文件名.jpg
//3.要求使用闭包方式完成
//4.strings.HasSuffix():判断字符串中是否包含后缀
package main
import (
"fmt"
"strings"
)
//闭包函数
func makeSuffix(suffix string) func (string) string {
//如果name没有指定后缀,则加上,否则返回原来的名字
return if !strings.HasSuffix(name string, suffix string) {
return name + suffix
}
return name
}
//传统函数
func makeSuffix2(suffix string, name string) string {
if strings.HasSuffix(name string, suffix string) {
return name + suffix
}
return name
}
func main() {
//测试闭包makeSuffix的使用
f2 := makeSuffix(".jpg)
fmt.Println("文件名处理后=",f2("winner")) //winner.jpg
fmt.Println("文件名处理后=",f2("test.avi")) //test.jpg
//传统makeSuffix2的使用
fmt.Println("文件名处理后=",makeSuffix2("jpg", "winner")) //winner.jpg
fmt.Println("文件名处理后=",makeSuffix2("pg", "test.avi")) //test.jpg
}
上面返回代码说明:
1.返回的匿名函数和makeSuffix(suffix string)的suffix变量组合成一个闭包,因为返回的函数引用到suffix这个变量
2.闭包的好处: 如果使用传统的方法,也可以轻松实现这个功能,但是传统的方法需要每次都传入后缀名,比如:.jpg,而闭包因为可以保留上次引用的某个值,所以我们传入一次就可以反复使用
二.defer
为什么需要defer
在函数中,程序员经常需要创建资源,比如:数据库连接,锁,文件句柄等,为了在函数执行完毕后,及时地释放资源,go的设计者提供defer(延时机制)
package main
import "fmt"
func sum(n1 int, n2 int) int {
//当执行到defer时,暂时不执行,会将defer后面的语句压入到独立的栈区(defer栈)
//当函数执行完毕后,再从defer栈,按照先入后出的方式出栈,执行
defer fmt.Println("n1 = ", n1) // defer 3. n1 = 10
defer fmt.Println("n2 = ", n2) //defer 2. n2 = 20
n3 := n1 + n2 //n3 = 30
frm.Println("n3 = ", n3) //1. n3 = 30
return n3
}
func main() {
res := sum(10, 20)
fmt.Println("res = ", res) //4.res = 30
}
defer细节说明:
1.当go执行到一个defer时,不会立即执行defer后面的语句,而是将defer后面的语句压入到一个栈中(暂时称为defer栈),然后继续执行函数下一个语句
2.当函数执行完毕后,再从defer栈中依次从栈顶取出执行(遵守先入后出原则)
3.在defer将语句放入到栈时,也会将相关的值拷贝同时入栈
//defer实践
func test() {
//关闭文件资源
file = openFile(文件名)
defer file.close()
//其它代码
}
func test1() {
//释放数据库资源
connect = openDatabase()
defer connect.close()
//其它代码
}
三.函数参数的传递方式
基本介绍
在讲解函数的注意事项和使用细节时,已经讲过值类型和引用类型,再在这里系统总结一下:值类型参数默认就是值传递,引用类型参数默认就是引用传递
两种传递方式
1).值传递
2).引用传递
其实,不管是值传递还是引用传递,传递给函数的都是变量的副本,不同的是,值传递的是值的拷贝,引用传递的是地址的拷贝.一般来说,地址拷贝的效率高,因为数据量小,而值拷贝决定拷贝的数据大小,数据越大,效率越低
值类型和引用类型
1).值类型: 基本数据类型系列:int系列,float系列,string,bool,数组和结构体(struct)
2.引用类型:指针,切片(slice),管道(channel),interface,map等
值传递和引用传递使用特点
1.值类型默认是值传递:变量直接存储值,内存通常分配在栈区
2.引用类型默认是引用传递:变量存储的是一个地址,这个地址对应的空间才是真正存储数据(值),内存通常在堆区分配,当没有任何变量引用这个地址时,该地址对应的数据空间就成为一个垃圾,由GC来回收
3.如果希望函数内的变量能修改函数外的变量,可以传入变量的地址&,函数内以指针的方式操作变量,从效果上看类似引用