Go 学习笔记 06 | 函数详解

一、Golang 函数

... 表示可变参数

package main

import (
	"fmt"
)


func sumFn1(x int, y ...int) int {
	fmt.Println(x, y)
	sum := x
	for _, v := range y {
		sum += v
	}
	return sum
}


func main() {
	sum1 := sumFn1(100, 1, 2, 3, 4)
	fmt.Println(sum1)
}

return 可以一次返回多个值

返回类型要一致,返回多个值的时候需要使用 ()

package main

import (
	"fmt"
)


func calc(x, y int) (int, int) {  // 返回多个值的时候需要使用 ()
	sum := x + y
	sub := x - y
	return sum, sub  // 返回类型要与上面 (int, int) 一致
}


func main() {
	a, b := calc(10, 2)
	fmt.Println(a, b)
}

返回值命名

函数定义时可以给返回值命名,并在函数体中直接使用这些变量,最后通过 return 关键字返回。

package main

import (
	"fmt"
)


func calc1(x, y int) (sum int, sub int) {  // 返回多个值的时候需要使用 ()
	fmt.Println(sum, sub)
	sum = x + y
	sub = x - y
	fmt.Println(sum, sub)
	return
}


func main() {
	a, b := calc1(10, 2)
	fmt.Println(a, b)
}

二、匿名函数

package main

import (
	"fmt"
)

func main() {
	// 匿名函数 匿名自执行函数
	func() {
		fmt.Println("test")
	}()

	// 匿名函数
	var fn = func(x, y int) int {
		return x * y
	}

	fmt.Println(fn(2, 3))

	// 匿名自执行函数接收参数
	func(x, y int) {
		fmt.Println(x + y)
	}(10, 20)
}

全局变量特点

  • 常驻内存
  • 污染全局

局部变量特点

  • 不常驻内存
  • 不污染全局

闭包

  • 可以让一个变量常驻内存
  • 可以让一个变量不污染全局
package main

import (
	"fmt"
)

// 闭包写法:函数里面嵌套一个函数,最后返回里面的函数
func adder1() func() int {
	var i = 10
	return func() int {
		return i + 1
	}
}

func adder2() func(y int) int {
	var i = 10
	return func(y int) int {
		i += y
		return i
	}
}


func main() {
	var fn1 = adder1()
	fmt.Println(fn1())
	fmt.Println(fn1())
	fmt.Println(fn1())

	var fn2 = adder2()
	fmt.Println(fn2(10))
	fmt.Println(fn2(10))
	fmt.Println(fn2(10))
}

输出:

10
10
10
20
30
40

defer

表示延迟执行函数。

package main

import (
	"fmt"
)

func f1() {
	fmt.Println("开始")

	defer func() {
		fmt.Println("aaa")
	}()  // 必须是匿名自执行函数,带()
	fmt.Println("结束")
}

func main(){
	f1()
}

输出:

开始
结束
aaa

函数中 return 语句底层实现:先返回值 = x,再 RET 指令。

deferreturn 语句底层实现:先返回值 = x,再运行 defer,最后 RET 指令。

package main

import (
	"fmt"
)

// 匿名返回值,操作变量不影响返回值
func f1() int{
	x := 5
	defer func() {
		x++
	}()  // 必须是匿名自执行函数,带()
	return x
}

// 命名返回值,返回值是操作以后的值
func f2() (x int){
	defer func() {
		x++
	}()  // 必须是匿名自执行函数,带()
	return 5
}

// 命名返回值
func f3() (y int){
	x := 5
	defer func() {
		x++
	}()  // 必须是匿名自执行函数,带()
	return x  // 返回的值相当于赋值给 y
}

// 命名返回值
func f4() (x int){  // x 一开始值为 0
	defer func(y int) {
		y++  // 这是一个局部变量
	}(x)  // defer 注册要延迟执行的函数时该函数所有的参数都需要确定其值
	return 5
}

func main(){
	fmt.Println(f1())
	fmt.Println(f2())
	fmt.Println(f3())
	fmt.Println(f4())
}

输出:

5
6
5
5

三、异常机制

Golang 目前(Go1.12)没有异常机制,但是使用 panic/recover 模式来处理错误。

panic 可以在任何地方引发,但是 recover 只有在 defer 调用的函数中有效。

package main

import (
	"fmt"
)

func fn1() {
	fmt.Println("fn1")
}

func fn2() {
	defer func() {
		err := recover()
		if err != nil{  // 有异常
			fmt.Println("err:", err)  // 抛出一个异常 是在这里打印出来的
		}
	}()
	panic("抛出一个异常")
}

func main() {
	fn1()
	fn2()
	fmt.Println("结束")
}

输出:

fn1
err: 抛出一个异常
结束

Golang 在实际中处理异常

package main

import (
	"fmt"
)

func fn1(a int, b int) int {
	defer func() {
		err := recover()
		if err != nil{  // 有异常
			fmt.Println("error:", err)  // error: runtime error: integer divide by zero
		}
	}()
	return a/b
}

func main() {
	fmt.Println(fn1(10, 0))
	fmt.Println("结束")
    fmt.Println(fn1(10, 2))
}

输出:

error: runtime error: integer divide by zero
0
结束
5

四、time 包

package main

import (
   "fmt"
   "time"
)

func main() {
   timeObj := time.Now()

   fmt.Println(timeObj)

   year := timeObj.Year()
   month := timeObj.Month()
   day := timeObj.Day()
   hour := timeObj.Hour()
   minute := timeObj.Minute()
   second := timeObj.Second()
   fmt.Printf("%d-%02d-%02d %02d:%02d:%02d", year, month, day, hour, minute, second)
}

输出:

2020-11-05 16:44:20.8303316 +0800 CST m=+0.001994801
2020-11-05 16:44:20

五、参考教程

Golang 教程 P24-P29

posted @ 2020-11-21 20:20  Wonz  阅读(105)  评论(0)    收藏  举报