golang复习

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

1. 利用defer、recover来实现try...catch

  1. func Try(fun func(),handler func(interface{})) {
  2. defer func() {
  3. if err := recover(); err != nil {
  4. handler(err)
  5. }
  6. }()
  7. fun()
  8. }
  9.  
  10. func main() {
  11. Try(func() {
  12. panic("foo")
  13. },func(e interface{}) {
  14. print(e)
  15. })
  16. }

2. 关于error的一个程序

error是一个类型,类似于string,error也可以定义自己的类型

  1. package main
  2.  
  3. import "errors"
  4. import "fmt"
  5.  
  6. // By convention,errors are the last return value and
  7. // have type `error`,a built-in interface.
  8. func f1(arg int) (int,error) {
  9. if arg == 42 {
  10.  
  11. // `errors.New` constructs a basic `error` value
  12. // with the given error message.
  13. return -1,errors.New("can't work with 42")
  14.  
  15. }
  16.  
  17. // A nil value in the error position indicates that
  18. // there was no error.
  19. return arg + 3,nil
  20. }
  21.  
  22. // It's possible to use custom types as `error`s by
  23. // implementing the `Error()` method on them. Here's a
  24. // variant on the example above that uses a custom type
  25. // to explicitly represent an argument error.
  26. type argError struct {
  27. arg int
  28. prob string
  29. }
  30.  
  31. func (e *argError) Error() string {
  32. return fmt.Sprintf("%d - %s",e.arg,e.prob)
  33. }
  34.  
  35. func f2(arg int) (int,error) {
  36. if arg == 42 {
  37.  
  38. // In this case we use `&argError` Syntax to build
  39. // a new struct,supplying values for the two
  40. // fields `arg` and `prob`.
  41. return -1,&argError{arg,"can't work with it"}
  42. }
  43. return arg + 3,nil
  44. }
  45.  
  46. func main() {
  47.  
  48. // The two loops below test out each of our
  49. // error-returning functions. Note that the use of an
  50. // inline error check on the `if` line is a common
  51. // idiom in Go code.
  52. for _,i := range []int{7,42} {
  53. if r,e := f1(i); e != nil {
  54. fmt.Println("f1 Failed:",e)
  55. } else {
  56. fmt.Println("f1 worked:",r)
  57. }
  58. }
  59. for _,e := f2(i); e != nil {
  60. fmt.Println("f2 Failed:",e)
  61. } else {
  62. fmt.Println("f2 worked:",r)
  63. }
  64. }
  65.  
  66. // If you want to programmatically use the data in
  67. // a custom error,you'll need to get the error as an
  68. // instance of the custom error type via type
  69. // assertion.
  70. _,e := f2(42)
  71. if ae,ok := e.(*argError); ok {
  72. fmt.Println(ae.arg)
  73. fmt.Println(ae.prob)
  74. }
  75. }

3. timer和ticker都是可以停止的

  1. package main
  2.  
  3. import (
  4. "fmt"
  5. "time"
  6. )
  7.  
  8. func main() {
  9. ticker := time.NewTicker(time.Millisecond * 500)
  10. go func() {
  11. for t := range ticker.C {
  12. fmt.Println("ticker is at ",t)
  13. }
  14. }()
  15.  
  16. time.Sleep(time.Millisecond * 1500)
  17. ticker.Stop()
  18. fmt.Println("ticker stopped")
  19. }
  1. package main
  2.  
  3. import (
  4. "fmt"
  5. "time"
  6. )
  7.  
  8. func main() {
  9. timer1 := time.NewTimer(time.Second * 2)
  10. <-timer1.C
  11. fmt.Println("timer1 expired.")
  12.  
  13. timer2 := time.NewTimer(time.Second * 1)
  14. go func() {
  15. <-timer2.C
  16. fmt.Println("timer2 expired.")
  17. }()
  18.  
  19. ok := timer2.Stop()
  20. if ok {
  21. fmt.Println("timer2 stopped.")
  22. }
  23. }

4. 一个比较复杂的channel的例子

  1. package main
  2.  
  3. import (
  4. "fmt"
  5. "math/rand"
  6. "sync/atomic"
  7. "time"
  8. )
  9.  
  10. type readOp struct {
  11. key int
  12. resp chan int
  13. }
  14. type writeOp struct {
  15. key int
  16. val int
  17. resp chan bool
  18. }
  19.  
  20. func main() {
  21. var ops int64 = 0
  22. reads := make(chan *readOp)
  23. writes := make(chan *writeOp)
  24.  
  25. go func() {
  26. var state = make(map[int]int)
  27. for {
  28. select {
  29. case read := <-reads:
  30. read.resp <- state[read.key]
  31. case write := <-writes:
  32. state[write.key] = write.val
  33. write.resp <- true
  34. }
  35. }
  36. }()
  37.  
  38. for r := 0; r < 100; r++ {
  39. go func() {
  40. for {
  41. read := &readOp{
  42. key: rand.Intn(5),resp: make(chan int)}
  43. reads <- read
  44. <-read.resp
  45. atomic.AddInt64(&ops,1)
  46. }
  47. }()
  48. }
  49.  
  50. for w := 0; w < 10; w++ {
  51. go func() {
  52. for {
  53. write := &writeOp{
  54. key: rand.Intn(5),val: rand.Intn(100),resp: make(chan bool)}
  55. writes <- write
  56. <-write.resp
  57. atomic.AddInt64(&ops,1)
  58. }
  59. }()
  60. }
  61.  
  62. time.Sleep(time.Second)
  63. opsFinal := atomic.LoadInt64(&ops)
  64. fmt.Println("ops:",opsFinal)
  65. }
5. sort包封装了一些常用的排序方法,用起来还是很方便的
  1. package main
  2.  
  3. import "fmt"
  4. import "sort"
  5.  
  6. func main() {
  7. strs := []string{"c","a","b"}
  8. sort.Strings(strs)
  9. fmt.Println("Strings:",strs)
  10.  
  11. ints := []int{7,2,4}
  12. sort.Ints(ints)
  13. fmt.Println("Ints: ",ints)
  14.  
  15. s := sort.IntsAreSorted(ints)
  16. fmt.Println("Sorted: ",s)
  17. }

6. slice的引用特性

  1. package main
  2.  
  3. import (
  4. "fmt"
  5. )
  6.  
  7. func main() {
  8. array := make([]int,3)
  9. array = append(array,1)
  10. a := array
  11. b := array
  12. a = append(a,2)
  13. b = append(b,3)
  14. fmt.Println(a)
  15. }




结果是什么呢?答案揭晓,输出是“[1 3]”。

就我的理解,slice 是一个{指向内存的指针,当前已有元素的长度,内存最大长度}的结构体,其中只有指向内存的指针一项是真正具有引用语义的域,另外两项都是每个 slice 自身的值。因此,对 slice 做赋值时,会出现两个 slice 指向同一块内存,但是又分别具有各自的元素长度和最大长度。程序里把 array 赋值给 a 和 b,所以 a 和 b 会同时指向 array 的内存,并各自保存一份当前元素长度 1 和最大长度 3。之后对 a 的追加操作,由于没有超出 a 的最大长度,因此只是把新值 2 追加到 a 指向的内存,并把 a 的“当前已有元素的长度”增加 1。之后对 b 进行追加操作时,因为 a 和 b 各自拥有各自的“当前已有元素的长度”,因此 b 的这个值依旧是 1,追加操作依旧写在 b 所指向内存的偏移为 1 的位置,也就复写了之前对 a 追加时写入的 2。

为了让 slice 具有引用语义,同时不增加 array 的实现负担,又不增加运行时的开销,似乎也只能忍受这个奇怪的语法了。

猜你在找的Go相关文章