一、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 指令。
带 defer
的 return
语句底层实现:先返回值 = 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
五、参考教程
有疑问加站长微信联系(非本文作者)