009. golang 函数 function

前端之家收集整理的这篇文章主要介绍了009. golang 函数 function前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

函数function

  • Go 函数 不支持 嵌套、重载和默认参数
  • 支持以下特性:

    1. 无需声明原型、不定长度变参、多返回值、命名返回值参数
    2. 匿名函数、闭包
  • 定义函数使用关键字 func,且左大括号不能另起一行

  • 函数也可以作为一种类型使用

  1. package main
  2.  
  3. import (
  4. "fmt"
  5. )
  6.  
  7. func main() {
  8. fmt.Println(A(1,"a"))
  9.  
  10. fmt.Println(B(2,"b"))
  11.  
  12. a,b,c := 1, 2, 3
  13. C("y",a,c)
  14. fmt.Println(a,c)
  15.  
  16. arr := []int{1, 3, 4, 5, 6}
  17. D(arr)
  18. fmt.Println(arr)
  19.  
  20. e := 2
  21. E(&e)
  22. fmt.Println(e)
  23.  
  24. }
  25.  
  26. func A(a int,b string) (int,string) {
  27. return a,b
  28. }
  29.  
  30. func B(a int,b string) (A int,B string) {
  31. A,B = a,b
  32. return
  33. }
  34.  
  35. //不定长变参,只能作为最后一个参数 而且拷贝的都是值拷贝
  36. func C(x string,a ...int) {
  37. a[0] = 4
  38. a[1] = 5
  39. a[2] = 6
  40. fmt.Println(x,a)
  41. }
  42.  
  43. //数组的话都是地址拷贝
  44. func D(a []int) {
  45. a[0] = 10
  46. a[1] = 11
  47. a[2] = 12
  48. fmt.Println(a)
  49. }
  50.  
  51. //如果 值拷贝 想更改为 地址拷贝
  52. //传参数的时候加上 &
  53. //方法形参加上 *
  54. func E(e *int) {
  55. *e = 9
  56. fmt.Println(*e)
  57. }

  1. package main
  2.  
  3. import (
  4. "fmt"
  5. )
  6.  
  7. func main() {
  8. //匿名函数
  9. a := func() {
  10. fmt.Println("Func A")
  11. }
  12. a()
  13.  
  14. fmt.Println()
  15.  
  16. //匿名函数+闭包
  17. f := test(10)
  18. fmt.Println("------------------------")
  19. fmt.Println(f(1))
  20. fmt.Println(f(2))
  21.  
  22. }
  23.  
  24. func test(x int) func(int) int {
  25. fmt.Printf("%p,%d\n",&x,x)
  26. return func(y int) int {
  27. fmt.Printf("%p,x)
  28. return x + y
  29. }
  30. }

defer

  • 执行方式类似其它语言中的析构函数,在函数体执行结束后
  • 按照调用顺序的相反顺序逐个执行
  • 即使函数发生严重错误也会执行
  • 支持匿名函数调用
  • 常用于资源清理文件关闭解锁以及记录时间等操作
  • 通过与匿名函数配合可在return之后修改函数计算结果
  • 如果函数体内某个变量作为defer时匿名函数的参数,则在定义defer
  • 时即已经获得了拷贝,否则则是引用某个变量的地址
  • Go 没有异常机制,但有 panic/recover 模式来处理错误
  • Panic 可以在任何地方引发,但recover只有在defer调用函数中有效,而且得在panic前面

  1. package main
  2.  
  3. import (
  4. "fmt"
  5. )
  6.  
  7. func main() {
  8. fmt.Println("a")
  9. defer fmt.Println("b")
  10. defer fmt.Println("c")
  11.  
  12. fmt.Println("------------------")
  13.  
  14. for i := 0; i < 3; i++ {
  15. defer func() {
  16. fmt.Println(i)
  17. }()
  18. }
  19. fmt.Println("---------------")
  20. }

  1. package main
  2.  
  3. import (
  4. "fmt"
  5. )
  6.  
  7. func main() {
  8. A()
  9. B()
  10. C()
  11. }
  12.  
  13. func A() {
  14. fmt.Println("func A")
  15. }
  16.  
  17. func B() {
  18. panic("panic in B")
  19. }
  20.  
  21. func C() {
  22. fmt.Println("func C")
  23. }

  1. package main
  2.  
  3. import (
  4. "fmt"
  5. )
  6.  
  7. func main() {
  8. A()
  9. B()
  10. C()
  11. }
  12.  
  13. func A() {
  14. fmt.Println("func A")
  15. }
  16.  
  17. func B() {
  18. defer func() {
  19. if err := recover(); err != nil {
  20. fmt.Println("recover in B")
  21. }
  22. }()
  23. panic("panic in B")
  24. }
  25.  
  26. func C() {
  27. fmt.Println("func C")
  28. }

  1. package main
  2.  
  3. import (
  4. "fmt"
  5. )
  6.  
  7. func main() {
  8. var fs = [4]func(){} //声明一个function类型的slice,长度为4
  9.  
  10. for i := 0; i < 4; i++ {
  11. defer fmt.Println("defer i= ",i) //这是一个i作为参数传进去的输出,因为i是int型,所以遵循一个规则值拷贝的传递,还有defer是倒序执行的,所以先后输出3,2,1,0,跟下面的defer交替执行4次
  12. defer func() { fmt.Println("defer_closure i = ",i) }() //执行完下面的代码后,到了该defer了,这也是一个匿名函数,同样的也没有参数,也没有定义i,所以这也是个闭包,用的也是外面的i,所以先输出4,接着执行上面的defer,这样反复执行4次
  13. fs[i] = func() { fmt.Println("closure i = ",i) } //把相应的4个匿名函数存到function类型的slice里,因为这是个匿名函数,又没有参数,且也没有定义i,所以i就是外层函数的地址引用,就是for循环的i的地址,执行完for后i的值为4,所以输出4个4
  14. }
  15.  
  16. for _,f := range fs { //用for循环对slice的调用,f为slice的值,即匿名函数,而f()则是对匿名函数调用
  17. f()
  18. }
  19. }

猜你在找的Go相关文章