07、Golang 教程 - Go 函数

  • 函数是基本的代码块,用于执行一个任务。
  • Go 中有 3 种函数:普通函数、匿名函数(没有名称的函数)、方法(定义在 struct 上的函数)。
  • Go 中函数参数可以没有名称,例如 func test(int,int)。
  • Go 中的函数可以作为一种 type 类型,例如 type callback func(int,int) int。
  • Go 程序中最少有一个 main() 函数。
  • Go 中不允许函数重载(overload),也就是说不允许函数同名。
  • Go 中的函数不能嵌套函数,但可以嵌套匿名函数(内联形式)。
  • 函数是一个值,可以将函数赋值给变量,使得这个变量也成为函数,例如 nextNumber := closePackage()。
  • 函数可以作为参数传递给另一个函数,函数的返回值可以是一个函数。这些特性使得函数变得无比的灵活,例如回调函数、闭包等等功能都依赖于这些特性。
  • Go 中的函数在 1.18 版本之前不支持泛型,但如果需要泛型的情况,大多数时候都可以通过 接口、type switch、reflection 的方式来解决。但使用这些技术使得代码变得更复杂,性能更低。
  • 可以通过函数来划分不同的功能,逻辑上每个函数执行的是指定的任务。
  • 函数声明告诉了编译器函数的名称,返回类型和参数。
  • Go 中的标准库提供了多种可动用的内置的函数。例如:len() 函数可以接受不同类型参数并返回该类型的长度。如果传入的是字符串则返回字符串的长度,如果传入的是数组,则返回数组中包含的元素个数。

1. 基础函数

语法

func 函数名(参数列表) (返回值列表) {
   
     
	......
}

无参数无返回值

func test() {
   
     
    ......
}

传参有返回值

func test(a int,b int) int {
   
     
    ......
    return n
}

传参有多个返回值

func test(a int,b int) (int,int) {
   
     
    ......
    return a+b,a*b
}

示例

// 定义 max() 函数传入两个整型参数 num1 和 num2,并返回两个参数的最大值
func max(num1,num2 int) int {
   
     
    if num1 > num2 {
   
     
        return num1
    } else {
   
     
        return num2
    }
}

// 函数的调用
package main

import "fmt"

func main() {
   
     
    var test1 int = 100
    var test2 int = 200
    var result int
    result = max(test1,test2)
    fmt.Println("最大值为:",result)
}

func max(num1,num2 int) int {
   
     
    if num1 > num2 {
   
     
        return num1
    } else {
   
     
        return num2
    }
}

// 函数返回多个值
package main

import "fmt"

func main() {
   
     
    a,b := multi_value(3,5)
    fmt.Println("和为:",a,"积为:",b)
}

func multi_value(num1,num2 int) (int,int) {
   
     
    result1 := num1 + num2
    result2 := num1 * num2
    return result1,result2
}
// 可以空白标识符剔除输出结果 _,b := multi_value(3,5)

2. 函数参数

  • 函数如果使用参数,该变量可称为函数的形参。
  • 形参就像定义在函数体内的局部变量。
  • 调用函数,可以通过两种方式来传递参数:
传递类型 描述
值传递 值传递是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数。
引用传递 引用传递是指在调用函数时将实际参数的地址传递到函数中,那么在函数中对参数所进行的修改,将影响到实际参数。
  • 默认情况下,Go 语言使用的是值传递,即在调用过程中不会影响到实际参数。

示例

// 使用引用类型进行两数交换
package main

import "fmt"

func main() {
   
     
	var (
		a = 10
		b = 20
	)
	fmt.Println("交换前:a=",a,"b=",b)
	swap(&a,&b)		// 交换了 a,b 的地址
	fmt.Println("交换后:a=",a,"b=",b)
}

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

// 结果
交换前:a= 10 b= 20
交换后:a= 20 b= 10

// 值传递示例
package main

import "fmt"

type Car struct {
   
     
	Price int
	Long int
}

func main() {
   
     
	c1 := new(Car)
	c1.Price = 200000
	c1.Long = 5
	fmt.Println("交换前车价:",c1.Price)
	fmt.Println("交换前车长:",c1.Long)
	swap(c1.Price,c1.Long)
	fmt.Println("交换后车价:",c1.Price)
	fmt.Println("交换后车长:",c1.Long)
}

func swap(a,b int) {
   
     
	a,b = b,a
}

// 结果
交换前车价: 200000
交换前车长: 5
交换后车价: 200000
交换后车长: 5

3. 函数作为实参

可以很灵活的创建函数,并作为另外一个函数的实参。

示例

package main

import (
    "fmt"
    "math"
)

func main() {
   
     
    /*声明函数变量*/
    getRoot := func(x float64) float64 {
   
     	// 匿名函数,没有给函数定义名称,赋值给函数变量
        return math.Sqrt(x)
    }
    fmt.Println(getRoot(9))	// 使用变量名调用函数
    
    /*math 数学包的使用*/
    fmt.Println("-10 的绝对值",math.Abs(float64(-10)))
    fmt.Println("5.2 向上取整",math.Ceil(5.2))
    fmt.Println("5.8 向下取整",math.Floor(5.8))
    fmt.Println("11 除以 3 的余数",math.Mod(11,3))
    fmt.Println(math.Modf(5.26))	//取整数,取小数
    fmt.Println("3 的 2 次方",math.Pow(3,2))
    fmt.Println("10 的 4 次方",math.Pow(10,4))
    fmt.Println("8 的开平方",math.Sqrt(8))
    fmt.Println("8 的开立方",math.Cbrt(8))
    fmt.Println("圆周率",math.Pi)
}

4. 回调函数

  • 和其他很多语言一样,golang 中函数也可以作为其它函数的参数进行传递,然后在其它函数内调用执行,一般称之为回调
  • Golang 是编译型语言,区别于解释型语言。解释型语言解释一条,编译一条,执行一条;golang 一个函数体一起编译,一起执行,如果函数体语句太多,其执行起来过于消耗计算机资源。所以我们需要实现程序间的解耦,节省计算机资源,提升运行效率。
  • 回调函数的使用可以大大提升编程的效率,这使得它在现代编程中被非常多地使用。同时,有一些需求必须要使用回调函数来实现。

示例一

package main

import "fmt"

// 声明函数类型,不写语句结构
type cback func(int) int

func main() {
   
     
    // 对回调函数进行隐匿,起到安全保护作用之余,提高程序运行效率
    test_cback(1,callback)
    test_cback(2,func(x int) int {
   
     
        fmt.Printf("回调:x:%d\n",x)
        return x
    })    
}

func test_cback(x int,f cback) {
   
     	// cback 函数作为参数传入,并起名为 f
    f(x)
}

func callback(x int) int {
   
     
    fmt.Printf("回调:x:%d\n",x)
    return x
}

// 结果
回调:x:1
回调:x:2

 

1、 main()调用初始函数test1();
2、 回调函数test2()是test1()的形参;
3、 test1()调用test2();
4、 test1()运行结束,返回main();

  • 函数可以作为另一个函数的参数(典型用法是回调函数)
  • 函数可以返回另一个函数,即让另一个函数作为这个函数的返回值(典型用法是闭包)
  • 函数可以作为一个值赋给变量(常用在匿名函数)

示例二

// 调用函数 test 时,调用真正的实现函数 add
package main

import "fmt"

type Callback func(x, y int) int

// 提供一个接口,让外部去实现
func test(x, y int, callback Callback) int {
   
     
    return callback(x, y)
}

// 回调函数的具体实现
func add(x, y int) int {
   
     
    return x + y
}

func main() {
   
     
    x, y := 1, 2
    fmt.Println(test(x, y, add))
} 

//结果
3

示例三

package main

/* 实现:加,减,乘,除 */

import "fmt"

type FuncType func(int, int) int

// 实现加法
func Add(a,b int) int {
   
     
	return a + b
}

// 实现减法
func Minus(a,b int) int {
   
     
	return a - b
}

// 实现乘法
func Mul(a,b int) int {
   
     
	return a * b
}

// 实现除法
func Div(a,b int) int {
   
     
	return a / b
}

// 回调函数:函数有一个参数是函数类型,这个函数就是回调函数
// 多态:多种形态,调用同一个接口,可以实现不同表现
func Calc(a,b int, test FuncType) (result int) {
   
     
	fmt.Println("Calc")
	result = test(a,b)
	return
}

func main() {
   
     
	x := Calc(4,2,Add)
	y := Calc(4,2,Minus)
	z := Calc(4,2,Mul)
	k := Calc(4,2,Div)
	fmt.Println("x =",x)
	fmt.Println("y =",y)
	fmt.Println("z =",z)
	fmt.Println("k =",k)
}

// 结果
Calc
Calc
Calc
Calc
x = 6
y = 2
z = 8
k = 2

5. 函数闭包

匿名函数可作为闭包。匿名函数是一个 “内联” 语句表达式。匿名函数的优越性在于可以直接使用函数内的变量,不必声明

在回调函数中,将函数作为传参。

闭包中,将函数作为返回值(return 返回函数,返回的函数再返回值)。

一般来说,回调函数和闭包函数附带的还具备一个特性:函数可以作为一个值赋值给变量。

Go 中函数不能嵌套命名函数,所以函数返回函数的时候,只能返回匿名函数(不能 func … func …)。

示例

package main

import "fmt"

// 包中包(函数中的函数),将匿名函数限制在 closePackage() 中
// 定义了闭包函数 closePackage() 和一个匿名函数(有一个返回值为 int 类型)
// 匿名函数可以继承 i := 0,不必声明,对匿名函数来说,这个变量相当于全局变量
// 在闭包函数中 return 匿名函数,并在匿名函数结构体中返回运算后 i 的值
func closePackage() func() int {
   
     
    i := 0
    // 返回,执行匿名函数,闭包结构
    return func() int {
   
     
        i += 1
        return i
    }
}

func main() {
   
     
    /*定义函数,使用闭包做 +1 操作*/
    nextNumber := closePackage()
    fmt.Println("使用 nextNumber 做自增")
    fmt.Println(nextNumber()) // 1
    fmt.Println(nextNumber()) // 2
    fmt.Println(nextNumber()) // 3
    fmt.Println("使用 nextNumber 做自增")
    nextNumber1 := closePackage()
    fmt.Println(nextNumber1()) // 1
    fmt.Println(nextNumber1()) // 2
}

// 结果
使用 nextNumber 做自增
1
2
3
使用 nextNumber 做自增
1
2

闭包= 函数 + 外层变量的引用
 
这一整个叫闭包结构。

6. 函数方法

同时有函数和方法。一个方法就是一个包含了接受者的函数,接受者可以是命名类型或者结构体类型的一个值或者是一个指针。所有给定类型的方法属于该类型的方法集。

语法:

func (var_name var_data_type) function_name() [return_type] {
   
     
    /*函数体*/
}

示例:

package main

import "fmt"

/*定义结构体*/
type Circle struct {
   
     
    radius float64
}

func main() {
   
     
    var c1 Circle
    c1.radius = 10.00
    fmt.Println("圆的面积 = ",c1.getArea())
}

// 该 method 属于 Circle 类型对象中的方法
func (c Circle) getArea() float64 {
   
     
    // c.radius 即为 Circle 类型对象中的属性
    return math.Pi * c.radius * c.radius
}	

// 结果
圆的面积 =  314.1592653589793

package main

import "fmt"

// 定义结构体作为接收者
type Car struct {
   
     
	// 属性
	Name string
	Color string
}

// 函数方法,这些函数方法属于结构体
func (c Car) Call() {
   
     	// (c Car) 相当于 c := new(Car)
	fmt.Printf("%s品牌的汽车,颜色是%s正在鸣笛",c.Name,c.Color)
}
func (c Car) Run() {
   
     
	fmt.Printf("%s品牌的汽车,颜色是%s正在行驶",c.Name,c.Color)
}

func main() {
   
     
	// 实例化 c1
	c1 := new(Car)
	c1.Name = "奔驰"
	c1.Color = "黑色"
	c1.Call()
	fmt.Println()
	// 实例化 c2
	c2 := new(Car)
	c2.Name = "宝马"
	c2.Color = "白色"
	c2.Run()
}

Go基础函数

[Go高阶函数][Go 1]