go函数详解

函数
学习好Go语言的函数,记住一个技巧:Go语言是强类型语言!

语法:

func 函数名(参数名 参数类型,…)(返回值类型1,返回值类型2,…){
return 返回值类型1,返回值类型2,…
}

func是定义函数的关键字,不可省略。

参数名后面要标明参数的类型,可以一个或多个参数,也可以是无参函数,连续相同类型的参数可以这样写:a,b int,其中a,b是变量名。

函数可以有一个或多个,也可以没有返回值,只需要写明返回的类型即可,在写return返回的时候要和返回值类型一一对应。单个返回值的时候可以去掉括号。
示例:

func main(){
   res := add(10,20) //30
}

func add(a,b int) int{
    return a + b
}

我们之前说过,Go语言是以包为单位管理的,它没有类和对象的概念,所以这并不同面向对象编程的思路以类管理。所以当你定义了一个方法时,这个方法在这个包内部(其他方法处)是可以直接拿定义名使用的,就像这样:
新了解:其实在Go语言中函数也是属于一个类型

func main(){
    var myfunc func() //定义一个函数类型
    myfunc = sayHello // add是在这个包内的方法,可以直接通过方法名赋值给我们定义的myfunc变量
    myfunc()
}

func sayHello(){
    fmt.Println("你好")
}

当我们了解到函数也属于类型的时候,就可以整活了,我们不妨设想试一试,能不能把函数作为参数,进行传递呢?

函数作为参数传递的话,我们需要写一个参数为函数的函数

func main(){
    res := alu(10,20,mult)
    fmt.Println(res)
}

// 首先定义一个函数,准备将它作为参数传递。该函数是一个乘法函数,接收两个float的数,然后将他们相乘并返回
func mult(a, b int) int {
	return a * b
}

// 定义一个函数用来接收函数,并调用这个函数(哈哈套娃行为)
func alu(a, b int, fun func(int, int) int) int {
	res := fun(a, b)
	return res
}

这样写的好处是我定义了一个对a和b通用操作的方法alu,而对a和b具体怎么操作,是看fun函数是怎么定义。

还有另外一种传参的方式:匿名函数

匿名函数
作为参数传递
还是依照上面的例子,这次我们使用匿名函数作为参数传递。那么啥是匿名函数?就是没有名字的函数

func main(){
    res = alu(10,20,func(a int, b int) int {
        return a * b
    })
    fmt.Println(res)
}

// 定义一个函数用来接收函数,并调用这个函数(哈哈套娃行为)
func alu(a, b int, fun func(int, int) int) int {
    res := fun(a, b)
    return res
}

注意哦,这里使用 res = 是因为我上个例子中 res 已经通过 :=定义过了,记住:=是定义用的。

没有名字的函数就别在外局单独定义啦,不然别人怎么找到呢?可以这样使用。在匿名函数的最后加上(传入的参数),实现自己调用执行,返回的结果可以用一个变量接收。

func main(){
    res = func(a int, b int) int {
        return a + b
    }(1, 2)
    fmt.Println(res) //打印匿名函数返回值
}

作为函数的返回值
作为函数的返回值,我们可以这样想,定义一个函数它的返回值是函数,通过一个例子感受一下

func main(){
    speed := fast_shoes("张三")
    fmt.Println(speed(10)) // 20
}

func fast_shoes(name string) func(_speed int) int {
    fmt.Println(name + "获得飞速鞋子一双")
    speed := 10
    return func(_speed int) int {
        _speed += speed
        return _speed
    }
}

fast_shoes函数,接收一个人的名字,返回的是一个函数(这个函数的返回值是int )。这个例子我们实现的是一个人获得了一双“飞速鞋子”,那么“飞速鞋子”的初始属性是10,我们要做到将这个人的初始速度(是10)再增加10。

fmt.Println(speed(10)),调用10次的结果输出也是20。那么如果我想说,我想持续一段时间内,这个鞋子还拥有越跑越快的效果呢?了解一下闭包

闭包
我们先对上述的代码改造一下:

func main(){
    speed := fast_shoes("张三")
    for i := 0; i < 10; i++ {
        fmt.Println(speed(10)) // 20,21,22,23,24,25,26,27,28,29
    }
}

func fast_shoes(name string) func(_speed int) int {
    fmt.Println(name + "获得飞速鞋子一双")
    speed := 10
    return func(_speed int) int {
        _speed += speed
        speed ++
        return _speed
    }
}

你会发现调用了10次,输出的结果再往上叠加。原因是我们加了speed++

我们称:内层函数+外层函数局部变量(可以是入参或者定义的局部变量) = 闭包结构

闭包结构有这样一个特点:speed局部变量不会随着外层函数fast_shoes的结束而销毁。本来执行完fast_shoes(“张三”)后,speed就会被销毁,但是此时内层函数使用了外层函数的局部变量时,speed这个局部变量的生命周期就发生改变了,它在main函数中的speed()不再调用时才会被销毁。