GO语言函数 | 青训营

104 阅读4分钟

Go 语言函数

1.函数特点

 无需声明原型。
    • 支持不定 变参。
    • 支持多返回值。
    • 支持命名返回参数。 
    • 支持匿名函数和闭包。
    • 函数也是一种类型,一个函数可以赋值给变量。

    • 不支持 嵌套 (nested) 一个包不能有两个名字一样的函数。
    • 不支持 重载 (overload) 
    • 不支持 默认参数 (default parameter)。

函数声明包含一个函数名,参数列表, 返回值列表和函数体。如果函数没有返回值,则返回列表可以省略。函数从第一条语句开始执行,直到执行return语句或者执行函数的最后一条语句。

函数可以没有参数或接受多个参数。

注意类型在变量名之后 。

当两个或多个连续的函数命名参数是同一类型,则除了最后一个类型之外,其他都可以省略。

函数可以返回任意数量的返回值。

使用关键字 func 定义函数,左大括号依旧不能另起一行。 有返回值的函数,必须有明确的终止语句,否则会引发编译错误。

你可能会偶尔遇到没有函数体的函数声明,这表示该函数不是以Go实现的。这样的声明定义了函数标识符

2.函数格式

在默认情况下,Go 语言使用的是值传递,即在调用过程中不会影响到实际参数。

注意1:无论是值传递,还是引用传递,传递给函数的都是变量的副本,不过,值传递是值的拷贝。引用传递是地址的拷贝,一般来说,地址拷贝更为高效。而值拷贝取决于拷贝的对象大小,对象越大,则性能越低。

注意2:map、slice、chan、指针、interface默认以引用的方式传递。

不定参数传值 就是函数的参数不是固定的,后面的类型是固定的。(可变参数)

Golang 可变参数本质上就是 slice。只能有一个,且必须是最后一个。

image-20230811180328140.png

image-20230811180339132.png

在go语言中使用地址传递会改变原来的值,使用值传递不会改变值的大小。

map、slice、chan、指针、interface默认以引用的方式传递。

image-20230811180621186.png Golang 可变参数本质上就是 slice。只能有一个,且必须是最后一个。

当然可以使用 args...interface{} 表示任意类型参数

image-20230811200929809.png 带返回值的函数

3.匿名函数

很像java中的匿名内部类 匿名函数是指不需要定义函数名的一种函数实现方式。1958年LISP首先采用匿名函数。

在Go里面,函数可以像普通变量一样被传递或使用,Go语言支持随时在代码里定义匿名函数。

匿名函数由一个不带函数名的函数声明和函数体组成。匿名函数的优越性在于可以直接使用函数内的变量,不必申明。

image-20230811201428834.png

最后返回3,这里将函数看做变量来进行操作,可以进行

4.延迟调用defer

image-20230811201620353.png

重点是先进后出

image-20230811203128004.png

image-20230811203139721.png

defer陷阱
defer 与 closure

image-20230811203810619.png

image-20230811203820383.png

defer 与 return

image-20230811203936854.png

输出结果是2,可以看出在返回值之前已经将i赋值为2

5.异常处理

Golang 没有结构化异常,使用 panic 抛出错误,recover 捕获错误

panic:

    1、内置函数
    2、假如函数F中书写了panic语句,会终止其后要执行的代码,在panic所在函数F内如果存在要执行的defer函数列表,按照defer的逆序执行
    3、返回函数F的调用者G,在G中,调用函数F语句之后的代码不会执行,假如函数G中存在要执行的defer函数列表,按照defer的逆序执行
    4、直到goroutine整个退出,并报告错误

recover:

    1、内置函数
    2、用来控制一个goroutine的panicking行为,捕获panic,从而影响应用的行为
    3、一般的调用建议
       a). 在defer函数中,通过recever来终止一个goroutine的panicking过程,从而恢复正常代码的执行
       b). 可以获取通过panic传递的error

注意:

    1.利用recover处理panic指令,defer 必须放在 panic 之前定义,另外 recover 只有在 defer 调用的函数中才有效。否则当panic时,recover无法捕获到panic,无法防止panic扩散。
    2.recover 处理异常后,逻辑并不会恢复到 panic 那个点去,函数跑到 defer 之后的那个点。
    3.多个 defer 会形成 defer 栈,后定义的 defer 语句会被最先调用。
package main
​
func main() {
    test()
}
​
func test() {
    defer func() {
        if err := recover(); err != nil {
            println(err.(string)) // 将 interface{} 转型为具体类型。
        }
    }()
​
    panic("panic error!")
}

6.自定义error

image-20230811210628415.png