Skip to content

Go语言函数

函数的定义

在Go语言中,使用关键字func定义函数。函数名和参数列表放在括号中,函数体放在花括号内。以下是一个简单的函数定义示例:

go
func add(x int, y int) int {
    return x + y
}

上面的代码定义了一个名为add的函数,它有两个参数xy,都是整数类型。函数的返回类型是int,它将两个参数相加并返回结果。

在Go语言中,可以将参数类型相同的参数合并到一个参数列表中。例如,上面的函数可以简化为:

go
func add(x, y int) int {
    return x + y
}

这里的xy都是整数类型,因此它们的类型可以在参数列表中省略。

函数的参数传递

在Go语言中,函数参数可以是值参数或引用参数。值参数是复制传递,而引用参数是传递指针。

例如,下面的代码定义了一个名为swap的函数,它交换两个整数类型的变量的值:

go
func swap(x, y int) {
    x, y = y, x
}

这里的xy都是值参数,因此函数体内的交换操作不会影响函数外部的变量。如果要在函数内部交换函数外部的变量,可以使用指针类型的参数。

例如,下面的代码定义了一个名为swapPtr的函数,它交换两个整数类型变量的值:

go
func swapPtr(xPtr, yPtr *int) {
    *xPtr, *yPtr = *yPtr, *xPtr
}

这里的xPtryPtr都是指向整数类型变量的指针。在函数体内,我们使用*操作符获取指针所指向的变量的值,并使用*操作符修改该变量的值。通过传递指针类型的参数,我们可以在函数内部交换函数外部的变量。

另外,Go语言还支持可变参数列表。可以使用...语法定义可变参数,例如:

go
func sum(nums ...int) int {
    total := 0
    for _, num := range nums {
        total += num
    }
    return total
}

这里的nums是可变参数列表,它可以接受任意数量的整数参数。在函数体内,我们使用range语法遍历可变参数列表,并累加所有参数的值。由于可变参数列表是切片类型,因此我们可以像处理切片一样处理它。

函数的返回值

在Go语言中,使用关键字return返回一个或多个值。如果函数没有返回值,则可以省略return语句。

例如,下面的代码定义了一个名为div的函数,它接受两个整数类型的参数,并返回它们的商和余数:

go
func div(x, y int) (int, int) {
    return x / y, x % y
}

这里的函数返回两个整数类型的值,它们分别是x / yx % y的结果。在函数体内,我们使用逗号分隔多个返回值。

如果函数只有一个返回值,可以省略括号。例如,下面的代码定义了一个名为double的函数,它接受一个整数类型的参数,并返回它的两倍:

go
func double(x int) int {
    return x * 2
}

这里的函数只返回一个整数类型的值,因此可以省略括号。

函数的作用域

在Go语言中,函数参数和局部变量只在函数内部可见。函数外部的代码无法访问函数内部的参数和局部变量。

例如,下面的代码定义了一个名为max的函数,它接受两个整数类型的参数,并返回它们的最大值:

go
func max(x, y int) int {
    if x > y {
        return x
    } else {
        return y
    }
}

这里的xy是函数的参数,它们只在函数内部可见。在函数体内,我们可以使用它们进行比较,并返回它们的最大值。

匿名函数

在Go语言中,可以定义匿名函数,也称为lambda函数。它们可以像普通函数一样被调用,并且可以被分配给变量。

例如,下面的代码定义了一个匿名函数,并将它分配给变量f

go
f := func(x, y int) int {
    return x + y
}

这里的匿名函数接受两个整数类型的参数,并返回它们的和。我们可以像调用普通函数一样调用它,例如:

go
result := f(3, 5)
fmt.Println(result) // 输出8

在上面的代码中,我们调用了变量f所代表的匿名函数,并传递了两个整数类型的参数。函数返回它们的和,并将结果赋给变量result

闭包

在Go语言中,函数可以访问其外部作用域中的变量。这种函数被称为闭包。闭包可以用于实现函数工厂。

例如,下面的代码定义了一个名为adder的函数,它返回一个函数,该函数可以将一个整数类型的值加到某个初始值上:

go
func adder(init int) func(int) int {
    sum := init
    return func(x int) int {
        sum += x
        return sum
    }
}

这里的adder函数返回一个函数,它接受一个整数类型的参数,并将它加到某个初始值上。在函数体内,我们使用闭包访问了外部作用域中的变量sum,并将它加上了函数参数x。每次调用返回的函数时,都会将sum的值更新为之前的值加上当前的参数值。

例如,下面的代码使用adder函数创建了两个可以累加整数类型值的函数:

go
add1 := adder(0)
add2 := adder(10)

fmt.Println(add1(1)) // 输出1
fmt.Println(add1(2)) // 输出3
fmt.Println(add2(1)) // 输出11
fmt.Println(add2(2)) // 输出13

在上面的代码中,我们首先使用adder函数创建了两个函数add1add2,它们的初始值分别是0和10。然后我们分别调用这两个函数,并传递不同的参数。由于add1add2是闭包,它们可以访问外部作用域中的变量sum,因此每次调用时都会将之前的值加上当前的参数值。

defer语句

在Go语言中,可以使用defer语句在函数返回之前执行一些代码。defer语句通常用于释放资源或记录日志。

例如,下面的代码定义了一个名为openFile的函数,它打开一个文件并返回文件句柄。在函数返回之前,我们使用defer语句关闭了文件句柄:

go
func openFile(filename string) (*os.File, error) {
    file, err := os.Open(filename)
    if err != nil {
        return nil, err
    }
    defer file.Close()
    return file, nil
}

这里的openFile函数打开一个文件,如果打开失败,则返回错误。如果打开成功,则使用defer语句在函数返回之前关闭文件句柄。这样可以确保文件句柄在函数返回之前被关闭,从而避免资源泄漏。

函数类型

在Go语言中,函数类型是一种值类型,可以将函数作为参数传递给其他函数或从函数中返回。

例如,下面的代码定义了一个名为apply的函数,它接受一个函数类型的参数和一个整数类型的参数,并将函数应用到该参数上:

go
func apply(f func(int) int, x int) int {
    return f(x)
}

这里的apply函数接受一个函数类型的参数f和一个整数类型的参数x,并将f应用到x上。在函数体内,我们调用了函数参数f,并传递了参数x。函数f必须接受一个整数类型的参数,并返回一个整数类型的值。

例如,下面的代码定义了一个名为double的函数,它接受一个整数类型的参数,并返回它的两倍:

go
func double(x int) int {
    return x * 2
}

然后我们可以使用apply函数将double函数应用到某个整数类型的值上,例如:

go
result := apply(double, 3)
fmt.Println(result) // 输出6

在上面的代码中,我们调用了apply函数,传递了double函数和整数类型的值3。apply函数将double函数应用到3上,并返回结果6。