Golang:3.4、Golang 函数

阅读 170

2022-10-20

3.4、Golang 函数

Go语言函数特性

  1. 函数分类:
  • 普通函数
  • 匿名函数
  • 方法
  1. 函数不能重载,即不允许函数同名
  2. 函数不能嵌套函数,但可以嵌套匿名函数
  3. 函数可以赋值给变量
  4. 函数可以作为参数传递给另一个函数
  5. 函数的返回值可以是一个函数
  6. 函数传参传递是参数的副本
  7. 函数参数可以没有名称

函数定义

func function_name([parameter list])[return_type] {
// 函数体
}

示例

package main

import fmt

func sum(a int, b int) (ret int) {
return a + b
}

func main() {
r := sum(1, 1)
fmt.Printf(%v, r)
// 2
}

函数返回值

没有返回值

func foo(){
fmt.Println(Hello)
}

一个返回值

func sum(a int, b int) (ret int) {
return a + b
}

多个返回值

func foo() (name string, age int) {
return Tom, 23
}

// 省略命名
func foo() (string, int) {
return Tom, 23
}

多值返回常用于函数返回错误

value, exists
value, ok
value, err

返回值过多(4个以上),通常放在容器中返回

同类型 slice
不同类型 map

返回值不想使用,可以使用 _ 丢弃

函数参数

参数类型:

  • 形参 声明函数时的参数列表
  • 实参 调用时传递的参数

特点:

  • 参数可以0个或者有多个,需要指定数据类型
  • 函数是传值的方式进行传参
  • 可以使用变长参数...

示例

package main

import fmt

// a, b形参
func sum(a int, b int) (ret int) {
return a + b
}

func main() {
// 1, 1 实参
r := sum(1, 1)
fmt.Printf(%v, r)
// 2
}

注意:

  • map、slice、interface、channel数据类型是指针,拷贝的是指针,有可能会改变原始数据
package main

import fmt

func foo(arr []int) {
arr[0] = 100
}

func main() {

a := []int{1, 2, 3}

foo(a)

fmt.Printf(%v, a)
// [100 2 3]
}

变长参数

package main

import fmt

func foo(args ...int) {
for _, value := range args {
fmt.Printf(%v , value)
}
}

func main() {

foo(1, 2, 3)

// 1 2 3
}

多种参数混合使用

package main

func foo(name string, isMan bool, arr ...int) {

}

func main() {

foo(Tom, true, 1, 2, 3)

}

函数类型与函数变量

定义函数类型

// 示例:接收两个参数,返回一个参数
type function_name func(int, int) int

示例

package main

import fmt

// 定义函数类型
type foo func(int, int) int

func sum(a int, b int) int {
return a + b
}

func max(a int, b int) int {
if a > b {
return a
} else {
return b
}
}

func main() {
// 声明
var f foo

f = sum
s := f(1, 2)
fmt.Printf(%v\n, s) // 3

f = max
m := f(1, 2)
fmt.Printf(%v\n, m) // 2

}

高阶函数

go语言的函数,可以作为函数的参数,也可以作为函数返回值

函数作为参数

package main

import fmt

func sayHello(name string) {
fmt.Printf(Hello %s, name)
}

func foo(name string, fun func(string)) {
fun(name)
}

func main() {

foo(Tom, sayHello)
// Hello Tom

}

函数作为返回值

package main

import fmt

func sum(a int, b int) int {
return a + b
}

func sub(a int, b int) int {
return a - b
}

func calc(name string) func(a int, b int) int {
switch name {
case +:
return sum
case -:
return sub
default:
return nil
}
}

func main() {
sum := calc(+)
r := sum(1, 1)
fmt.Printf(%v, r)
// 2
}

匿名函数

语法格式

// 没有函数名称
func (参数列表)(返回值){

}

示例1

package main

import fmt

func main() {
sum := func(a int, b int) int {
return a + b
}

ret := sum(1, 1)
fmt.Printf(%v, ret)
// 2
}

示例2

package main

import fmt

func main() {
ret := func(a int, b int) int {
return a + b
}(1, 1)

fmt.Printf(%v, ret)
// 2
}

闭包

闭包可以理解成:定义在一个函数内部的函数,本质上是将函数内部和函数外部连接起来的桥梁

闭包 = 函数 + 引用环境

示例1

package main

import (
fmt
)

// 返回一个函数
func add() func(int) int {
var a int
return func(b int) int {
a += b
return a
}
}

func main() {
f := add()
fmt.Println(f(1)) // 1
fmt.Println(f(1)) // 2
fmt.Println(f(1)) // 3

f2 := add()
fmt.Println(f2(1)) // 1
fmt.Println(f2(1)) // 2
fmt.Println(f2(1)) // 3
}

示例2

package main

import (
fmt
strings
)

// 返回一个函数
func makeSuffixFunc(suffix string) func(string) string {
return func(name string) string {
if strings.HasSuffix(name, suffix) {
return name
} else {
return name + suffix
}
}
}

func main() {
txtFunc := makeSuffixFunc(.txt)
jpgFunc := makeSuffixFunc(.jpg)

fmt.Println(txtFunc(test)) // test.txt
fmt.Println(jpgFunc(test)) // test.jpg
}

示例3

package main

import (
fmt
)

// 返回两个函数
func calc(base int) (func(int) int, func(int) int) {

add := func(i int) int {
return base + i
}

sub := func(i int) int {
return base - i
}

return add, sub
}

func main() {
f1, f2 := calc(10)
fmt.Println(f1(1), f2(2))
// 11 8
}

递归

递归函数:函数内部调用函数自身的函数

递归函数特点

  • 递归就是自己调用自己
  • 必须定义退出条件,否则就会成为死循环
  • 递归很可能会出现栈空间内存溢出

示例:阶乘

for循环实现

package main

import fmt

// 阶乘
func foo(n int) int {
ret := 1
for i := 1; i <= n; i++ {
ret *= i
}

return ret
}

func main() {
// 5! = 5 x 4 x 3 x 2 x 1
ret := foo(5)
fmt.Println(ret)
// 120
}

递归实现

package main

import fmt

// 阶乘
func foo(n int) int {

if n == 1 {
// 退出条件
return 1
} else {
// 自己调用自己
return n * foo(n-1)
}
}

func main() {
// 5! = 5 x 4 x 3 x 2 x 1
ret := foo(5)
fmt.Println(ret)
// 120
}

菲波那切数列

计算公式

f(n) = f(n-1) + f(n-2) 

f(2) = f(1) = 1

代码实现

package main

import fmt

// 菲波那切数列
func foo(n int) int {
if n == 1 || n == 2 {
// 退出条件
return 1
} else {
// 递归表达式
return foo(n-1) + foo(n-2)
}
}

func main() {
// foo(5) = foo(4) + foo(3) = 3 + 2 = 5
// foo(4) = foo(3) + foo(2) = 2 + 1 = 3
// foo(3) = foo(2) + foo(1) = 1 + 1 = 2
// foo(2) = 1
// foo(1) = 1

ret := foo(5)
fmt.Println(ret)
// 5
}

defer语句

defer语句后面的语句延迟处理,在defer归属的函数即将返回时,将延迟处理的语句按照defer定义的逆序进行执行

也就是说,先定义的defer语句最后执行;后定义的defer语句,先执行

defer特性

  • 关键字defer用于注册延迟调用
  • 这些调用知道return前才被执行,可以用来做资源清理
  • 多个defer语句,按照先进后出的顺序执行
  • defer语句中的变量,defer声明时就决定了

defer用途

  • 关闭文件句柄
  • 锁资源释放
  • 数据库连接释放

示例

package main

import fmt

func main() {
fmt.Println(start)

defer fmt.Println(defer1)
defer fmt.Println(defer2)
defer fmt.Println(defer3)

fmt.Println(end)

// start
// end
// defer3
// defer2
// defer1
}

init函数

init函数是特殊函数,会先于main函数执行,实现包级别的初始化操作

init函数特点

  • init函数先于main函数自动执行,不能被其他函数调用
  • init函数没有输入参数,没有返回值
  • 每个包可以有多个init函数
  • 包的每个源文件也可以有多个init函数
  • 同一个包的init执行顺序,golang没有明确意义
  • 不同胞的init函数按照包导入的依赖关系决定执行顺序

golang初始化顺序

变量初始化 -> init() -> main()

示例

package main

import fmt

var i int = initVar()

func initVar() int {
fmt.Println(initVar)
return 100
}

func init() {
fmt.Println(init)
}

func main() {
fmt.Println(main)
}

// initVar
// init
// main

精彩评论(0)

0 0 举报