GO学习之 函数(Function)

GO系列

1、GO学习之Hello World

2、GO学习之入门语法

3、GO学习之切片操作

4、GO学习之 Map 操作

5、GO学习之 结构体 操作

6、GO学习之 通道(Channel)

7、GO学习之 多线程(goroutine)

8、GO学习之 函数(Function)

文章目录

前言

按照公司目前的任务,go 学习是必经之路了,虽然行业卷,不过技多不压身,依旧努力!!!

之前在 结构体 篇中提到了方法,那方法其实和函数是差不多的,不过方法与函数的区别是,函数不属于任何类型,方法属于特定的类型,这句话在 结构体 篇中也说到了,但函数的使用上也有许多细节需要注意的,此篇则给予详解。

一、什么是函数?

  • 在 Go 语言中,函数(Function)是一种可执行的代码块(对特定功能进行提取,形成代码片段),用于执行特定的任务或操作。
  • 函数是 Go 语言中的基本组件,实现了模块化和复用的机制,让代码更加结构化和可维护。

下面列举了我能查到和想到的特点(不仅限于这些):

Go语言中函数的特点:

  • 无需声明原型
  • 支持不定参数
  • 支持多返回值
  • 支持命名返回参数
  • 支持匿名函数和闭包
  • 函数也是一种类型,可以把一个函数赋值给一个变量
  • 不能嵌套定义函数
  • 不能像JAVA中那样重载函数(overload)

二、函数声明

  • 函数的声明使用 关键字 func
  • 基本语法:func 函数名(参数列表) 返回值列表 { 函数体 }
  • 函数名命名规范:1、最好驼峰命名,见名知意,比如:addNum(a,b int){};2、首字母不能是数字;3、首字母大写表示可以被本包和其他包文件使用,类似 public,比如:AddNum(a,b int){},首字母小写则类似 private,比如:addNum(a,b int){}
  • 参数列表用逗号分隔,每个参数有参数名和类型组成,比如:func list(pageNo int, pageSize int) (int, error) { }
  • 如果多个参数是同一类型,则前面参数类型可以省略,比如:func list(pageNo, pageSize int) (int, error) { }
  • 使用 ... 语法可以为函数定义可变参数,运行函数接受不定数量参数,比如:``
  • 如果是无放回值则可省略,比如:func save(id int, name string) { }
  • 如果是一个返回值则不需要(或有都可)小括号,比如:func save(id int, name string) int { }
  • 如果是多个返回值则用小括号包起来,并且和 return 语句一 一对应,比如:func save(id int, name string) (int, string) { ...return 1,'success' }
  • 上面提到支持命名返回函数,比如:func divideAndRemainder(a, b int) (quotient, remainder int) { }
  • 使用关键词 func 定义函数,大括号不能另起一行

上代码:

三、函数调用

  • 在接受函数返回值时,如果多个返回值则一一对应接受,比如:count, result := save(1, "张三")
  • 如果只需要接受其中一个返回值,另一个不需要接受,则可以用下划线 _ 忽略,比如:count, _ := save(1, "张三")
  • 如果 main 包中想调用其他包中的函数,那其他包中的函数则需要定义为包外可访问的,函数名首字母大写,比如定义一个 func1 的包:func SumNum(a, b int) int { },此时需要注意此包名,必须使用包名调用,比如:s := func1.SumNum(1, 2)

下面例子,举例了 Go 语言常用到的函数定义的例子,可供参考。

例子是哥我一个一个敲的,测试通过,并且附上了运行结果,不过只靠眼睛看还是有点繁杂,还是自己在专门敲一遍为好,不过对于大佬无所谓了,对于像我这种小白,则只能按部就班敲一遍,解决各种报错,方能成长!

包路径是这样的:

go 复制代码
package func1

import "fmt"

// 定义无参数无返回值函数
func test() {
	fmt.Println("call test函数")
}

// 定义有参数无返回值函数,此函数私有的,只有内部可调
func addNum(a, b int) {
	c := a + b
	fmt.Printf("a + b = c %+v\n", c)
}

// 定义有参数有一个返回值函数, 次函数共有的,内部、外部包均可调
func SumNum(a, b int) int {
	c := a + b
	return c
}

// 定义可变参数函数
func ParamsFunc(params ...string) {
	for index, item := range params {
		fmt.Printf("可变参数为 %d:%+v\n", index, item)
	}
}

// 定义有参数有多个返回值函数
func List(pageNo, pageSize int) (int, []string) {
	fmt.Printf("查询操作...%d, %d", pageNo, pageSize)
	result := []string{"特斯拉", "广汽", "丰田", "宝马", "奥迪"}
	return 5, result
}

// 定义命名返回函数
func divideAndRemainder(a, b int) (quotient, remainder int) {
	quotient = a / b
	remainder = a % b
	return // 省略了 return 语句,并且直接返回了命名的返回值变量
}

下面示例是对上面定义的函数进行调用。

主要,import 其他包的路径是 gotest.com/test/src/functionTest/func1

go 复制代码
package main

import (
	"fmt"

	"gotest.com/test/src/functionTest/func1"
)

func main() {
	// 调用本包中的 save 函数,接受两个返回值
	count1, result := save(1, "张三")
	fmt.Printf("接受 save 函数的两个返回值 count1:%+v, result: %v\n", count1, result)

	// 调用本包中的 save 函数,接受一个返回值
	count, _ := save(1, "张三")
	fmt.Printf("接受 save 函数的一个返回值 count: %+v\n", count)

	// 调用无返回值函数
	list2(1, 10)

	// 调用 func1 包中的 SumNum 函数
	s := func1.SumNum(1, 2)
	fmt.Printf("调用 func1 包中的 SunNum 函数结果:%+v\n", s)
	
		// 调用可变参数函数
	func1.ParamsFunc("特斯拉", "广汽", "丰田", "宝马", "奥迪")

	// 调用 func1 包中的 List 函数
	totalCount, carBrands := func1.List(1, 10)
	fmt.Printf("调用 func1 包中的 List 函数,查询结果:%+v 条,数据:%v\n", totalCount, carBrands)
}

// 定义有参数有多个返回值函数
func save(id int, name string) (int, string) {
	fmt.Printf("保存%+v,%v\n", id, name)
	return 1, "success"
}

// 定义有多个参数无返回值函数
func list2(pageNo, pageSize int) {
	fmt.Println("list 接口")
}

运行结果:

bash 复制代码
PS D:\workspaceGo\src\functionTest\main> go run funcTest.go
保存1,张三
接受 save 函数的两个返回值 count1:1, result: success
保存1,张三
接受 save 函数的一个返回值 count: 1
list 接口
调用 func1 包中的 SunNum 函数结果:3
可变参数为 0:特斯拉
可变参数为 1:广汽
可变参数为 2:丰田
可变参数为 3:宝马
可变参数为 4:奥迪
查询操作...1, 10调用 func1 包中的 List 函数,查询结果:5 条,数据:[特斯拉 广汽 丰田 宝马 奥迪]

四、匿名函数

  • 在 Go 语言中,支持匿名函数,也就是没有函数名的函数
  • 可以将匿名函数赋值给变量
  • 也可以将匿名函数直接调用,则是闭包
go 复制代码
package main

import "fmt"

func main() {
	// 定义匿名函数直接调用
	func() {
		fmt.Println("匿名函数调用!")
	}()
	// 定义匿名函数赋值给变量 hello
	hello := func() {
		fmt.Println("Hello 函数调用!")
	}
	// 调用匿名函数
	hello()
	// 定义有参数的匿名函数
	sum := func(a, b int) int {
		return a + b
	}
	fmt.Printf("加法计算:%+v\n", sum(1, 2))
}

运行结果:

go 复制代码
PS D:\workspaceGo\src\functionTest\main> go run .\anonymousFunc.go
匿名函数调用!
Hello 函数调用!
加法计算:3

下面是一个稍微复杂点的例子:

下面例子中,我们把 函数 作为一个成员存放在了 数组 fns、结构体 s、管道 fc 中,并且获取到函数进行调用。

go 复制代码
package main

func main() {
	// 定义数据,元素类型是一个函数
	fns := [](func(a int) int){func(a int) int { return a + 1 }, func(a int) int { return a + 2 }}
	// 获取数组中的第一个函数调用,传参 10
	for _, fn := range fns {
		println(fn(10))
	}

	// 定义一个结构体,成员是一个 函数,调用结构体的 函数成员
	s := struct {
		fn func() string
	}{
		fn: func() string { return "Hello World!" },
	}
	println(s.fn())

	// 定义一个管道,发送一个函数,再接受到函数进行调用
	fc := make(chan func() string, 2)
	fc <- func() string { return "fc: Hello World!" }
	println((<-fc)())
}

运行结果:

bash 复制代码
PS D:\workspaceGo\src\functionTest\main> go run .\anomymousFunc2.go
11
12
Hello World!
fc: Hello World!

五、函数参数和返回值

  • 在 Go 语言中,函数可以作为参数传递,也可以作为另一个函数的返回值

下面是一个比较简单的示例,例子中接受一个 函数类型参数 fc,返回一个 匿名函数。

go 复制代码
package func1

import "fmt"

func CallFunc(fc func()) func() {
	fmt.Println("接受到函数 fc, 开始回调!")
	// 返回一个匿名函数
	return func() {
		fc()
		fmt.Println("call back...")
	}
}

调用代码:

go 复制代码
package main

import (
	"fmt"

	"gotest.com/test/src/functionTest/func1"
)

func main() {
fc := func() {
		fmt.Println("我是参数 fc 执行!")
	}
	fr := func1.CallFunc(fc)
	fr()
}

运行结果:

go 复制代码
PS D:\workspaceGo\src\functionTest\main> go run .\funcTest.go
接受到函数 fc, 开始回调!
我是参数 fc 执行!
call back...

下面是 ChatGPT给出的经典案例,方便更加深入理解函数如何作为参数和返回值在实际场景中的应用,示例我已测试,ojbk。

  • 函数作为参数使用:
go 复制代码
package main

import "fmt"

// 函数类型作为参数
type MathFunc func(int, int) int

// 加法函数
func add(a, b int) int {
	return a + b
}

// 减法函数
func subtract(a, b int) int {
	return a - b
}

// 计算函数,接收一个函数类型参数,并执行该函数
func calculate(a, b int, op MathFunc) int {
	return op(a, b)
}

func main() {
	// 调用 calculate 函数,传入 add 函数作为参数
	result := calculate(10, 5, add)
	fmt.Println("加法结果:", result)

	// 调用 calculate 函数,传入 subtract 函数作为参数
	result = calculate(10, 5, subtract)
	fmt.Println("减法结果:", result)
}

运行结果:

bash 复制代码
PS D:\workspaceGo\src\functionTest\main> go run .\gptFunc.go
加法结果: 15
减法结果: 5
  • 函数作为返回值使用:
go 复制代码
package main

import "fmt"

// 返回一个加法函数
func getAddFunc() func(int, int) int {
	// 返回一个匿名函数,来实现计算
	return func(a, b int) int {
		return a + b
	}
}

// 返回一个减法函数
func getSubtractFunc() func(int, int) int {
	return func(a, b int) int {
		return a - b
	}
}

func main() {
	// 获取加法函数并调用
	addFunc := getAddFunc()
	result := addFunc(10, 5)
	fmt.Println("加法结果:", result)

	// 获取减法函数并调用
	subtractFunc := getSubtractFunc()
	result = subtractFunc(10, 5)
	fmt.Println("减法结果:", result)
}

执行结果:

bash 复制代码
PS D:\workspaceGo\src\functionTest\main> go run .\gptFunc2.go
加法结果: 15
减法结果: 5

六、延迟执行函数

defer 的特性:

  • 关键字 defer 用户注册延迟调用,比如:defer println(i)
  • 注册的延迟调用直到 return 前才会执行,所以很适合做关闭、资源回收等操作
  • defer 语句,是按照先进后出的方式执行
  • defer 语句中的变量,在 defer 声明是就决定了

defer 适用场景:

  • 关闭流操作
  • 资源释放
  • 数据库连接释放
  • 等...

6.1 defer 先进后出

go 复制代码
package main

func main() {
	arr := [5]int{1, 2, 3, 4, 5}
	for i := range arr {
		defer println(i)
	}
}

运行结果:

bash 复制代码
PS D:\workspaceGo\src\functionTest\main> go run .\deferTest.go
4
3
2
1
0

从结果可以看出,先循环到的 defer 等到后面才执行。

6.2 defer 闭包函数

go 复制代码
package main

func main() {
	arr := [5]int{1, 2, 3, 4, 5}
	for i := range arr {
		defer func() {
			println(i)
		}()
	}
}

运行结果:

bash 复制代码
PS D:\workspaceGo\src\functionTest\main> go run .\deferTest.go
4
4
4
4
4

为啥全部变为了 4,由于循环体内的是闭包函数,声明完之后立马执行,但是在函数声明的时候 i 变量已经变为了 4,所以 4 个匿名函数都输出了 4。

由于 defer 看起来情况比较多,所以请移步到这里!

七、错误处理

  • Go 语言中的多数函数会返回一个错误 error 作为额外的返回值,用户表示函数是否执行成功
  • 调用函数通常需要检查错误,以便根据情况进行处理
  • 对于无返回值的函数,可以使用错误类型来表明函数是否执行成功,或者用 panic 来触发异常

7.1 使用 error 作为返回参数

在示例中,我们使用错误类型 error 来表示函数是否执行成功,如果函数出现错误,则返回 error。

go 复制代码
package main

import (
	"errors"
	"fmt"
)

func main() {
	err := divide(10, 0)
	if err != nil {
		fmt.Println("发生异常:", err)
	}
}

func divide(a, b int) error {
	if b == 0 {
		return errors.New("参数不能为 0")
	}
	return nil
}

运行结果:

bash 复制代码
PS D:\workspaceGo\src\functionTest\main> go run .\errorFunc.go
发生异常: 参数不能为 0

7.2 使用 panic 触发异常

示例中,用 panic 来触发异常表示函数执行状态,当函数出现错误时,直接触发异常,并中断程序执行。

**注意:**我们用 recover() 函数来捕获并处理异常,避免程序崩溃。recover 函数只在 defer 块中才有效,所以在 main() 函数中使用 defer 来捕获异常。

go 复制代码
package main

func main() {
	defer func() {
		if r := recover(); r != nil {
			println("发生异常:", r)
		}
	}()
	divide2(10, 0)
}

func divide2(a, b int) {
	if b == 0 {
		panic("参数不能为 0")
	}
}

运行结果:

bash 复制代码
PS D:\workspaceGo\src\functionTest\main> go run .\panicFunc.go
发生异常: (0xff1920,0x1011638)

八、总结

函数是 Go 语言中非常重要的组成部分,它们提供了模块化、代码复用和抽象的能力。通过函数,我们可以将复杂的逻辑划分为多个小模块,使得代码更加清晰、可读性更强,并且更易于维护和扩展。函数的灵活性和多样性使得 Go 语言可以用于解决各种不同的问题和场景。

相关推荐
tyler_download34 分钟前
golang 实现比特币内核:实现基于椭圆曲线的数字签名和验证
开发语言·数据库·golang
hlsd#35 分钟前
go mod 依赖管理
开发语言·后端·golang
杜杜的man39 分钟前
【go从零单排】迭代器(Iterators)
开发语言·算法·golang
亦世凡华、40 分钟前
【启程Golang之旅】从零开始构建可扩展的微服务架构
开发语言·经验分享·后端·golang
Chrikk10 小时前
Go-性能调优实战案例
开发语言·后端·golang
幼儿园老大*10 小时前
Go的环境搭建以及GoLand安装教程
开发语言·经验分享·后端·golang·go
canyuemanyue10 小时前
go语言连续监控事件并回调处理
开发语言·后端·golang
杜杜的man10 小时前
【go从零单排】go语言中的指针
开发语言·后端·golang
有梦想的咸鱼_14 小时前
go实现并发安全hashtable 拉链法
开发语言·golang·哈希算法
杜杜的man18 小时前
【go从零单排】go中的结构体struct和method
开发语言·后端·golang