Go 函数详解 func 匿名函数 闭包

简介: 函数是任何一门编程语言最重要的组成部分之一。函数简单理解是一段代码的封装:把一段逻辑抽象出来封装到一个函数中,给他取个名字,每次需要的时候调用这个函数即可。使用函数能够让代码更清晰,更简洁。

函数是任何一门编程语言最重要的组成部分之一。函数简单理解是一段代码的封装:把一段逻辑抽象出来封装到一个函数中,给他取个名字,每次需要的时候调用这个函数即可。使用函数能够让代码更清晰,更简洁。


定义


下面的代码段介绍了go语言中函数定义的各种情况,以及延迟函数的使用。


package main
import "fmt"
// 函数的定义
func f1(x int, y int) (ret int) {
   return x + y
}
// 无返回值的函数
func f2(x int, y int) {
   fmt.Println(x + y)
}
// 没有参数也没有返回值的
func f3() {
   fmt.Println("1111")
}
// 没有参数 有返回值的
func f4() int {
   return 4
}
// 返回值可以命名也可以不命名
// 命名的返回值就相当于在函数中声明一个变量
func f5(x int, y int) (ret int) {
   ret = x + y  //注意:因为已经在返回值中声明了ret,所以这里用= 而不是:= ,避免重复声明问题
   return //因为已经在函数体中声明了ret,所以在return的时候不需要重复声明
}
// 多个返回值
func f6() (int, int) {
   return 1, 2
}
// 多个参数简写的方式
// 当参数的类型一致时,可以将连续的相同参数 前面参数的类型省略 比如:
func f7(x, y, z int, a, b string, c, d bool) int {
   return x + y + z
}
// 可变长参数
// 可变长参数必须放在函数参数的最后
func f8(x string, y ...int) {
   fmt.Println(x)
   fmt.Println(y)
}
// defer 延迟执行
func deferDemo() {
   defer fmt.Println("111") //最先defer的语句最后执行
   defer fmt.Println("222")
   fmt.Println("333")
}
// go语言中函数没有默认参数的概念
func main() {
   r := f5(1, 2)
   fmt.Println(r)
   m, n := f6()
   fmt.Println(m, n)
   r7 := f7(1, 2, 3, "1", "1", true, false)
   fmt.Println(r7)
   f8("hah") //可变长度 不填也可以
   f8("hah", 1, 2, 3, 4)
   //延迟函数测试
   deferDemo()
}


函数也可以作为函数的参数


举个栗子:


package main
import "fmt"
func f2() int {
   return 2
}
// 函数也可以作为函数参数的类型
func f3(x func() int) {
   ret := x()
   fmt.Printf("f3打印ret的值:%v\n", ret)  //2
   fmt.Printf("f3打印ret的类型:%T\n", ret) //int
}
func main() {
   a := f2
   fmt.Printf("a的类型:%T\n", a)
   f3(a)
}


打印结果:


微信图片_20221112161147.jpg


函数作为函数的返回值


package main
import "fmt"
func f2() int {
   return 2
}
func ff(x, y int) int {
   return x + y
}
// 函数不仅可以作为参数,还可以作为返回值
func f5(x func() int) func(int, int) int {
   return ff
}
func main() {
   f7 := f5(f2)
   fmt.Printf("f7的值:%v\n",f7) //f7返回的是一个函数
   fmt.Printf("f7的类型:%T\n", f7)
}


打印结果:


微信图片_20221112161156.jpg


总结:


  1. 我们打印f7的值是一个内存地址
  2. f7的类型和我们预期的一致,返回了函数类型,就是我们定义的ff()函数


匿名函数


匿名函数就是没有名字的函数。匿名函数多用于实现回调函数和闭包。

在Go语言中函数内部不能再像之前那样定义函数了,只能定义匿名函数。


匿名函数的定义格式如下:


func(参数)(返回值){
    函数体
}


匿名函数因为没有函数名,所以没办法像普通函数那样调用,所以匿名函数需要保存到某个变量或者作为立即执行函数:


func main() {
  // 将匿名函数保存到变量
  add := func(x, y int) {
    fmt.Println(x + y)
  }
  add(10, 20) // 通过变量调用匿名函数
  //自执行函数:匿名函数定义完加()直接执行
  func(x, y int) {
    fmt.Println(x + y)
  }(10, 20)
}


总结:自执行函数就是在匿名函数后面追加(),表示不需要外部调用,直接执行。


闭包


闭包是一个函数,这个函数包含了他外部作用域的一个变量


举个栗子


package main
import "fmt"
func adder(x int) func(int) int {
   return func(y int) int {
      x += y
      return x
   }
}
func main() {
   f1 := adder(1)
   ret := f1(2)
   fmt.Println(ret)
}


打印结果:

微信图片_20221112161159.jpg


总结: 上面的栗子就是一个典型的闭包结构:匿名函数内部包含了他外部的变量x。


闭包=函数+引用环境


总结


这篇文章我们详细介绍了Go语言中函数的定义、也介绍了匿名函数和闭包的知识点。


公众号:程序员升级打怪之旅

微信号:wangzhongyang1993

B站视频:王中阳Go

相关文章
|
5月前
|
存储 运维 安全
go语言中闭包与匿名函数是什么?
本文探讨了Go语言中的匿名函数与闭包。首先介绍了匿名函数的定义与使用方式,包括直接调用、赋值给变量以及作为全局变量的应用。接着深入解析了闭包的概念及其本质,强调闭包能实现状态保持,但也警告其不当使用可能导致复杂的内存管理和运维问题。通过示例展示了如何利用闭包实现累加器功能,并对比了使用结构体字段的方法。最后,通过一个并发场景的示例说明了闭包在Go中处理多协程安全访问共享数据的应用,展示了闭包结合锁机制确保数据一致性的方式。
|
2月前
|
JSON 安全 网络协议
go语言使用内置函数和标准库
【10月更文挑战第18天】
21 3
|
4月前
|
Go
go函数
go函数
37 10
|
4月前
|
编译器 Go C++
Go to Learn Go之函数
Go to Learn Go之函数
34 0
|
4月前
|
编译器 Go 索引
Go数组、多维数组和切片(动态数组),及常用函数len(),cap(),copy(),append()在切片中的使用
本文介绍了Go语言中数组、多维数组和切片(动态数组)的基本概念和操作,包括数组的定义、初始化、访问,多维数组的定义和访问,以及切片的创建、使用和扩容。同时,还讲解了切片中常用的函数len()、cap()、copy()和append()的使用方法。
|
5月前
|
Prometheus Cloud Native Go
Go 1.22 标准库 slices 新增函数和一些旧函数增加新特性
Go 1.22 标准库 slices 新增函数和一些旧函数增加新特性
|
5月前
|
安全 编译器 Go
Go 1.21: 泛型函数的全面回顾
Go 1.21: 泛型函数的全面回顾
|
5月前
|
Go
深入理解 Go 中的 new() 和 make() 函数
深入理解 Go 中的 new() 和 make() 函数
|
5月前
|
设计模式 Java 数据库连接
|
5月前
|
Go 开发者