上下文包最佳实践

在Go中使用上下文包有几个最佳实践:

  • 使用context.WithCancel,context.WithTimeout或context.WithDeadline创建带有超时或取消信号的上下文。
  • 始终将上下文作为第一个参数传递给可能需要很长时间才能完成的函数,例如网络请求或数据库查询。
  • 使用context.Value存储和检索与上下文关联的值,例如用户ID或请求ID。
  • 使用context.WithValue基于现有上下文创建新上下文,并将其他值与其关联。
  • 检查上下文的Done通道,查看它是否已被取消。
  • 在整个应用程序中使用上下文包来传播请求范围的值和取消信号,而不是使用全局变量或手动信号。
  • 避免使用context.Background(),因为它没有超时或取消信号,而是使用context.TODO() 表示上下文稍后将被调用方替换。
  • 不要将上下文存储在结构中,而是将它们作为参数传递给函数。
  • 始终检查上下文感知函数的错误返回值,以查看上下文是否被取消或超时。

 

Go中的上下文包用于跨API边界传递请求范围的值、取消信号和截止时间。它可用于存储元数据、取消信号、超时和其他请求范围的值。上下文包提供了一种取消长时间运行的操作以及跨API边界存储元数据的方法。它通常与http包一起使用,以管理HTTP请求的请求范围值和取消信号。

它允许您在多个函数调用和goroutine之间传播请求范围的值,从而更轻松地管理应用程序中的信息流。

context.Context是使用context.With*函数创建,例如context.WithValuecontext.WithCancelcontext.WithTimeout。这些函数返回新的上下文,携带指定值或信号的上下文值。

context.Context可以作为参数传递给需要访问请求范围值或侦听取消信号的函数和方法。然后,这些函数可以使用context.Value和context.Done访问上下文中存储的值和信号的done方法。

在需要取消长时间运行的操作或跨多个goroutine传播请求范围的值的情况下,上下文包特别有用。它通常用于服务器端编程和其他并发方案。

应始终将上下文作为第一个参数传递给执行可能被取消的工作的任何函数。

例如,HTTP服务器可以使用上下文在客户端断开连接时取消请求的工作,数据库包可以使用上下文来实现可取消的查询,等等。

context包定义了 Context 类型,该类型是一个Go接口,具有四个方法,分别名为Deadline()、Done()、Err() 和Value():

type Context interface {
    Deadline() (deadline time.Time, ok bool)
    Done() 
    Err() 
    Value(key) 
}

上下文接口定义的方法

方法描述
Value(key)返回对应的值
Done()此方法返回可用于接收取消通知的频道
Deadline()此方法返回time.Time, 表示请求的截止日期,如果没有指定截止日期,则布尔值为false。
Err()此方法返回一个错误,指示完成通道接收信号的原因。上下文包定义了两个可用于比较错误的变量:Canceled表示请求已取消,DeadlineExeeded表示截止日期已过。

用于创建上下文值的上下文包函数

方法描述
Background()此方法返回默认上下文,从中派生其他上下文。
WithCancel(ctx)此方法返回一个上下文和一个取消函数。
WithDeadline(ctx, time)此方法返回一个带有截止日期的上下文,该截止日期用time.Time表示值。
WithTimeout(ctx, duration)此方法返回一个带有截止日期的上下文,该截止日期用time.Time表示值。
WithValue(ctx, key, val)此方法返回一个包含指定键值对的上下文。

 

context.WithCancel

下面是一个示例,说明context.WithCancel可以在Go中使用:

package main

    import (
        "context"
        "fmt"
        "time"
    )
    
    func doWork(ctx context.Context) {
        for {
            select {
            case <-ctx.Done():
                fmt.Println("Work done!")
                return
            default:
                fmt.Println("Working...")
            }
        }
    }
    
    func main() {
        ctx, cancel := context.WithCancel(context.Background())
        defer cancel() // 推迟执行cancel
    
        go doWork(ctx)
    
        // Wait for a while before canceling the context
        select {
        	case <-ctx.Done():
        	case <-time.After(time.Second * 3):
            	cancel()
        }
    }

在此示例中,我们使用context创建一个新的上下文 ctx.WithCancel(context.background())中。context.Background() 函数返回一个空的context。context.WithCancel 返回一个新的上下文和一个取消函数。我们推迟取消函数,以便在主函数退出时调用它。在 doWork 函数中,它将检查上下文是否已完成,如果是,则返回该函数。

在main函数中,我们正在运行一个goroutine,并通过传递上下文来完成其中的工作。等待3秒后,main函数会通过调用cancel函数来取消上下文,这将使上下文的Done通道关闭。因此,doWork函数将从Done通道接收,打印 "Work done!" 并返回。

 

context.WithTimeout

在Go中,您可以使用context.WithTimeout函数创建一个新上下文,该上下文在指定的超时时间过后被取消。该函数采用两个参数:现有上下文和超时持续时间。

下面是如何使用上下文的示例WithTimeout创建在5秒后取消的上下文:

package main

    import (
        "context"
        "fmt"
        "time"
    )
    
    func main() {
        ctx := context.Background()
        ctx, cancel := context.WithTimeout(ctx, 5*time.Second)
        defer cancel()
    
        // Do some work
        select {
        case <-ctx.Done():
            fmt.Println("Work completed")
        case <-time.After(10 * time.Second):
            fmt.Println("Work took longer than 10 seconds")
        }
    }

在此示例中,上下文是通过调用context.WithTimeout与后台上下文5秒的超时创建。该函数返回一个新上下文和一个用于取消上下文的函数。cancel函数在defer语句中调用,以确保在函数返回时取消上下文。select语句用于等待上下文完成或超时结束。

您还可以使用ctx检查上下文是否已完成。Done() 通道,您可以在select语句或循环中使用此通道来检查上下文是否完成,如果完成则意味着上下文已过期。

 

context.WithDeadline

在Go中,context.WithDeadline 函数创建具有关联截止时间的新上下文。截止日期是一个特定的时间点,在此时间点之后,上下文将被视为"死亡",任何相关工作都将被取消。该函数接受两个参数:现有上下文和截止时间。它返回一个新的上下文,该上下文将在指定的截止时间取消。

下面是一个示例:

package main

    import (
        "context"
        "fmt"
        "time"
    )
    
    func main() {
        ctx := context.Background()
        deadline := time.Now().Add(time.Second * 5)
        ctx, cancel := context.WithDeadline(ctx, deadline)
        defer cancel()
    
        select {
        case <-time.After(time.Second * 10):
            fmt.Println("overslept")
        case <-ctx.Done():
            fmt.Println(ctx.Err())
        }
    }

在此示例中,将创建一个背景上下文,然后设置未来5秒的截止时间。WithDeadline 函数用于基于背景上下文创建具有指定截止时间的新上下文。select语句用于等待上下文被取消或等待10秒过去。如果在10秒之前取消上下文,它将打印错误消息上下文截止时间超过,否则将打印"overslept"

 

使用Context的SQL查询超时

要在Golang中使用超时的SQL查询,您可以使用上下文包来设置查询执行的截止时间。首先,使用上下文创建具有超时的context.WithTimeout函数。然后,将上下文作为第一个参数传递给查询执行函数(例如db.QueryContext() 或db.ExecContext())。

下面是如何为SELECT查询设置1秒超时的示例:

package main

    import (
      "context"
      "database/sql"
      "fmt"
      "time"
    )
    
    func main() {
      // Open a connection to the database
      db, _ := sql.Open("driverName", "dataSourceName")
    
      // Create a context with a timeout of 1 second
      ctx, cancel := context.WithTimeout(context.Background(), time.Second)
      defer cancel()
    
      // Execute the query with the context
      rows, err := db.QueryContext(ctx, "SELECT * FROM table")
      if err != nil {
        fmt.Println(err)
      }
      defer rows.Close()
    
      // Handle the query results
      // ...
    }

 

使用上下文超时读取文件

在Go中,可以使用上下文包来设置读取文件的超时时间。下面是一个示例:

package main

    import (
        "context"
        "fmt"
        "io/ioutil"
        "time"
    )
    
    func main() {
        ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second)
        defer cancel()
    
        data, err := ioutil.ReadFile("example.txt", ctx)
        if err != nil {
            fmt.Println("Error:", err)
            return
        }
    
        fmt.Println(string(data))
    }

在此示例中,我们首先使用context创建一个超时为2秒的context.WithTimeout。然后,我们将此上下文传递给ioutil。ReadFile读取文件"example.txt"的内容。如果读取文件的时间超过2秒,则上下文的Done通道将关闭,并且ioutil.ReadFile返回错误。

 

将上下文用于HTTP

在Go中,您可以使用上下文包来设置HTTP请求的超时。下面是一个示例:

package main

    import (
        "context"
        "fmt"
        "io/ioutil"
        "net/http"
        "time"
    )
    
    func main() {
        ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second)
        defer cancel()
    
        req, err := http.NewRequest("GET", "https://example.com", nil)
        if err != nil {
            fmt.Println("Error:", err)
            return
        }
        req = req.WithContext(ctx)
    
        client := &http.Client{}
        resp, err := client.Do(req)
        if err != nil {
            fmt.Println("Error:", err)
            return
        }
        defer resp.Body.Close()
    
        body, err := ioutil.ReadAll(resp.Body)
        if err != nil {
            fmt.Println("Error:", err)
            return
        }
    
        fmt.Println(string(body))
    }

在此示例中,我们首先使用context创建一个超时为2秒的context.WithTimeout。然后,我们使用req.WithContext(ctx) 将此上下文附加到HTTP请求中。当我们使用http.Client.Do方法发出请求时,如果在收到响应之前关闭了上下文的Done通道,它将自动取消请求。

您还可以使用客户端库(如golang.org/x/net/context/ctxhttp)发出带有上下文的http请求,该库具有Get和Post方法,该方法将上下文作为第一个参数并返回响应和错误。

package main

  import (
      "context"
      "fmt"
      "io/ioutil"
      "net/http"
      "time"
  
      "golang.org/x/net/context/ctxhttp"
  )
  
  func main() {
      ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second)
      defer cancel()
  
      resp, err := ctxhttp.Get(ctx, nil, "https://example.com")
      if err != nil {
          fmt.Println("Error:", err)
          return
      }
      defer resp.Body.Close()
  
      body, err := ioutil.ReadAll(resp.Body)
      if err != nil {
          fmt.Println("Error:", err)
          return
      }
  
      fmt.Println(string(body))
  }

此示例使用ctxhttp.Get方法向"https://example.com"发出GET请求,超时为2秒。

请务必注意,在这两种情况下,如果上下文关闭了Done通道,则请求将被取消,但不会关闭连接。应用程序负责关闭连接。

 

使用Context作为键值存储

在Go中,您可以使用上下文包来存储可以与请求或一段代码一起传递的键值数据对。这允许您将其他信息与请求或代码段相关联,而不必将其作为显式参数传递。下面是一个示例:

package main

    import (
        "context"
        "fmt"
    )
    
    func main() {
        ctx := context.WithValue(context.Background(), "user_id", "12345")
        // use the context in a function
        processRequest(ctx)
    }
    
    func processRequest(ctx context.Context) {
        userID := ctx.Value("user_id").(string)
        fmt.Println("User ID:", userID)
    }

在此示例中,我们首先使用context创建一个上下文。WithValue方法。我们传递上下文。Background() 作为父上下文,以及方法的键值对 "user_id" 和 "12345"。然后,我们将此上下文传递给processRequest函数。在函数中,我们使用Value方法从上下文中检索"user_id"值,并将其打印出来。

请务必注意,上下文值仅用于传输进程和API边界的请求范围数据,而不用于将可选参数传递给函数。如果需要将可选参数传递给函数,最好使用结构或函数选项模式。

此外,上下文值不是线程安全的,如果您处于并发环境中,请使用sync.Map或同等产品。

© 本文著作权归作者所有。转载请联系授权,禁止商用。

🔗 系列文章

1. Go语言教程之边写边学:Go语言介绍

2. Go语言教程之边写边学:变量

3. Go语言教程之边写边学:常量

4. Go语言教程之边写边:数据类型定义

5. Go语言教程之边写边学:类型转换-如何在Go中将字符串转换为整数类型?

6. Go语言教程之边写边学:类型转换-如何在Go中将字符串转换为浮点类型?

7. Go语言教程之边写边学:类型转换-如何在Go中将字符串转换为布尔数据类型转换?

8. Go语言教程之边写边学:类型转换-如何在Go中将布尔类型转换为字符串?

9. Go语言教程之边写边学:类型转换-如何在 Go 中将浮点转换为字符串类型?

10. Go语言教程之边写边学:类型转换-integer到string转换的不同方式

11. Go语言教程之边写边学:类型转换-将Int数据类型转换为 Int16 Int32 Int64

12. Go语言教程之边写边学:类型转换-将float32转换为float64,将float64转换为float32

13. Go语言教程之边写边学:类型转换-将integer转换为float

14. Go语言教程之边写边学:运算符

15. Go语言教程之边写边学:If...Else...Else If条件语句

16. Go语言教程之边写边学:Switch…Case条件语句

17. Go语言教程之边写边学:For循环

18. Go语言教程之边写边学:函数

19. Go语言教程之边写边学:可变参数函数

20. Go语言教程之边写边学:延迟调用函数

21. Go语言教程之边写边学:panic和recover

22. Go语言教程之边写边学:数组array

23. Go语言教程之边写边学:切片slice

24. Go语言教程之边写边学:映射map

25. Go语言教程之边写边学:结构 struct

26. Go语言教程之边写边学:接口 interface

27. Go语言教程之边写边学:协程 Goroutine

28. Go语言教程之边写边学:通道 channel

29. Go语言教程之边写边学:并发 Concurrency

30. Go语言教程之边写边学:日志 logging

31. Go语言教程之边写边学:操作文件和文件夹,压缩和解压缩

32. Go语言教程之边写边学:读写不同的文件类型

33. Go语言教程之边写边学:正则表达式 Regex

34. Go语言教程之边写边学:golang中处理DNS记录

35. Go语言教程之边写边学:密码学 Cryptography

36. Go语言教程之边写边学:golang的一些陷阱

37. Go语言教程之边写边学:导入导出:如何从其他包或者子包中导入结构体

38. Go语言教程之边写边学:导入导出:用指针访问并修改其他包的变量

39. Go语言教程之边写边学:导入导出:如何导入包并声明包的别名

40. Go语言教程之边写边学:导入导出:如何实现来自不同包的接口?

41. Go语言教程之边写边学:导入导出:如何从另一个包调用函数?

42. Go语言教程之边写边学:导入导出:解引用来自另一个包的指针

43. Go语言教程之边写边学:常用的软件库:结构体和字段验证

44. Go语言教程之边写边学:常用的软件库:Golang中的动态JSON

45. Go语言教程之边写边学:常用的软件库:Golang统计包

46. Go语言教程之边写边学:常用的软件库:slice和map过滤器

47. Go语言教程之边写边学:常用的软件库:HTML解析器

48. Go语言教程之边写边学:常用的软件库:常用正则表达式包CommonRegex

49. Go语言教程之边写边学:常用的软件库:简单图像处理包

50. Go语言教程之边写边学:常用的软件库:图表包

51. Go语言教程之边写边学:常用的软件库:动态XML解析器

52. Go语言教程之边写边学:常用的软件库:时间工具包

53. Go语言教程之边写边学:web应用:如何创建照片库

54. Go语言教程之边写边学:web应用:获取Twitter趋势某个位置附近的热门主题标签

55. Go语言教程之边写边学:web应用:生成二维码的Web应用程序

56. Go语言教程之边写边学:web应用:读取和写入JSON数据的Web应用程序

57. Go语言教程之边写边学:Goroutines和Channels练习:启动多个Goroutine,每个goroutine都向一个频道添加值

58. Go语言教程之边写边学:Goroutines和Channels练习:从通道发送和接收值

59. Go语言教程之边写边学:Goroutines和Channels练习:将斐波那契数列读写到通道

60. Go语言教程之边写边学:Goroutines和Channels练习:Goroutines通道执行顺序

61. Go语言教程之边写边学:Goroutines和Channels练习:查找奇数和偶数

62. Go语言教程之边写边学:Goroutines和Channels练习:哲学家就餐问题

63. Go语言教程之边写边学:Goroutines和Channels练习:检查点同步问题

64. Go语言教程之边写边学:Goroutines和Channels练习:生产者消费者问题

65. Go语言教程之边写边学:Goroutines和Channels练习:睡眠理发师问题

66. Go语言教程之边写边学:Goroutines和Channels练习:吸烟者问题

67. Go语言教程之边写边学:Golang中的反射:Reflect 包的copy函数

68. Go语言教程之边写边学:Golang中的反射:Reflect包的DeepEqual函数

69. Go语言教程之边写边学:Golang中的反射:Reflect包的swapper函数

70. Go语言教程之边写边学:Golang中的反射:Reflect包的TypeOf函数

71. Go语言教程之边写边学:Golang中的反射:Reflect包的ValueOf函数

72. Go语言教程之边写边学:Golang中的反射:Reflect包的Field相关函数

73. Go语言教程之边写边学:Golang中的反射:Reflect包的Make相关函数

74. Go语言教程之边写边学:golang中创建结构体切片

75. Go语言教程之边写边学:golang中创建结构体字典

76. Go语言教程之边写边学:golang中捕获panic

77. Go语言教程之边写边学:检查结构体中是否存在某个字段

78. Go语言教程之边写边学:初始化包含结构体切片的结构体

79. Go语言教程之边写边学:使用空接口动态添加结构体成员

80. Go语言教程之边写边学:将结构字段转换为映射字符串

81. Go语言教程之边写边学:Golang中的字符串

82. Go语言教程之边写边学:Golang中的字符串:字符串操作

83. Go语言教程之边写边学:Golang中的字符串:字符串字符编码

84. Go语言教程之边写边学:Golang Web服务器示例

85. Go语言教程之边写边学:Go中的HTTP服务器

86. Go语言教程之边写边学:Go中的HTTP客户端

87. Go语言教程之边写边学:使用HTTP客户端在HTTP请求中设置标头

88. Go语言教程之边写边学:使用HTTP客户端从HTTP响应中读取标头

89. Go语言教程之边写边学:处理HTTP重定向

90. Go语言教程之边写边学:如何处理HTTP 错误

91. Go语言教程之边写边学:如何在HTTP响应中设置标头

92. Go语言教程之边写边学:如何读取HTTP响应中的标头

93. Go语言教程之边写边学:如何在HTTP请求中设置cookie

94. Go语言教程之边写边学:如何读取HTTP请求中的cookie

95. Go语言教程之边写边学:如何处理HTTP身份验证

96. Go语言教程之边写边学:如何在HTTP客户端处理HTTP超时

97. Go语言教程之边写边学:如何优雅地处理HTTP服务器关闭

98. Go语言教程之边写边学:如何处理HTTP客户端-服务器安全

99. Go语言教程之边写边学:如何处理HTTP服务器负载均衡

100. Go语言教程之边写边学:如何处理HTTP服务器缓存

101. Go语言教程之边写边学:如何处理HTTP客户端缓存

102. Go语言教程之边写边学:如何处理HTTP服务器健康检查

103. Go语言教程之边写边学:如何处理HTTP客户端请求压缩

104. Go语言教程之边写边学:如何处理HTTP服务器日志记录

105. Go语言教程之边写边学:如何创建HTTP/2服务器

106. Go语言教程之边写边学:如何向Prometheus发出服务器预警

107. Go语言教程之边写边学:标准库:Context包

108. Go语言教程之边写边学:基础练习:打印金字塔型星号

109. Go语言教程之边写边学:基础练习:检查一个数字是否是回文

110. Go语言教程之边写边学:基础练习:打印乘法表

111. Go语言教程之边写边学:基础练习:打印帕斯卡三角形

112. Go语言教程之边写边学:基础练习:如何创建多个goroutine,如何使用三个逻辑处理器

113. Go语言教程之边写边学:基础练习:如何提高 Golang 应用程序的性能?

114. Go语言教程之边写边学:基础练习:类型嵌入和方法覆盖的接口示例

115. Go语言教程之边写边学:基础练习:如何使用WaitGroup将main函数的执行延迟到所有goroutines完成后

116. Go语言教程之边写边学:基础练习:逐行读取文件到字符串

117. Go语言教程之边写边学:基础练习:并发最佳实践

118. Go语言教程之边写边学:基础练习:2023年要遵循的最佳实践

119. Go语言教程之边写边学:基础练习:如何获取带有本地时区的当前日期和时间

120. Go语言教程之边写边学:基础练习:获取今天是星期几,今天是今年的第几天,本周是今年的第几周?

121. Go语言教程之边写边学:基础练习:获取 EST、UTC 和 MST 中的当前日期和时间

122. Go语言教程之边写边学:基础练习:获取两个日期之间的小时、天、分钟和秒差 [未来和过去]

123. Go语言教程之边写边学:基础练习:将年、月、日、小时、分钟、秒、毫秒、微秒和纳秒添加到当前日期时间

124. Go语言教程之边写边学:基础练习:将年、月、日、小时、分钟、秒、毫秒、微秒和纳秒从当前日期时间减去

125. Go语言教程之边写边学:基础练习:以各种格式获取当前日期和时间

126. Go语言教程之边写边学:基础练习:从当前日期和时间获取年、月、日、小时、分钟和秒

127. Go语言教程之边写边学:基础练习:将特定的UTC日期时间转换为 PST、HST、MST 和 SGT

128. Go语言教程之边写边学:基础练习:使用carbon日期时间包

129. Go语言教程之边写边学:基础练习:切片排序、反转、搜索功能

130. Go语言教程之边写边学:基础练习:常用字符串函数(1)

131. Go语言教程之边写边学:基础练习:常用字符串函数(2)

132. Go语言教程之边写边学:基础练习:常用字符串函数(3)

133. Go语言教程之边写边学:基础练习:常用字符串函数(4)

134. Go语言教程之边写边学:基础练习:方法和对象

135. Go语言教程之边写边学:基础练习:接口的设计理念

136. Go语言教程之边写边学:数据结构与算法:线性搜索

137. Go语言教程之边写边学:数据结构与算法:二分搜索

138. Go语言教程之边写边学:数据结构与算法:插值搜索

139. Go语言教程之边写边学:数据结构与算法:冒泡排序

140. Go语言教程之边写边学:数据结构与算法:快速排序

141. Go语言教程之边写边学:数据结构与算法:选择排序

142. Go语言教程之边写边学:数据结构与算法:希尔排序

143. Go语言教程之边写边学:数据结构与算法:插入排序

144. Go语言教程之边写边学:数据结构与算法:梳排序

145. Go语言教程之边写边学:数据结构与算法:归并排序

146. Go语言教程之边写边学:数据结构与算法:基数排序

147. Go语言教程之边写边学:数据结构与算法:烧饼排序

148. Go语言教程之边写边学:数据结构与算法:二叉树

149. Go语言教程之边写边学:数据结构与算法:Rabin-Karp

150. Go语言教程之边写边学:数据结构与算法:链表 Linked List

151. Go语言教程之边写边学:数据结构与算法:LIFO堆栈和FIFO队列

152. Go语言教程之边写边学:数据结构与算法:BFPRT中位数的中位数

153. Go语言教程之边写边学:数据结构与算法:LCS最长公共子序列

154. Go语言教程之边写边学:数据结构与算法:Levenshtein Distance编辑距离

155. Go语言教程之边写边学:数据结构与算法:KMP算法 字符串匹配算法

156. Go语言教程之边写边学:数据结构与算法:Floyd–Warshall多源最短路径

157. Go语言教程之边写边学:数据结构与算法:汉诺塔Tower of Hanoi

158. Go语言教程之边写边学:数据结构与算法:哈夫曼编码(Huffman Coding)

159. Go语言教程之边写边学:数据结构与算法:绘制长方体

160. Go语言教程之边写边学:数据结构与算法:生成随机迷宫

161. Go语言教程之边写边学:数据结构与算法:生成数字折线矩阵

162. Go语言教程之边写边学:数据结构与算法:生成数字螺旋矩阵

163. Go语言教程之边写边学:数据结构与算法:生成自平衡二叉查找树 AVL tree

164. Go语言教程之边写边学:数据结构与算法:打印给定字符串的所有排列

165. Go语言教程之边写边学:数据结构与算法:LZW 数据无损压缩和解压缩

166. Go语言教程之边写边学:了解go中并发工作原理:了解goroutine

167. Go语言教程之边写边学:了解go中并发工作原理:将channel用作通信机制

168. Go语言教程之边写边学:了解go中并发工作原理:了解有缓冲channel

169. Go语言教程之边写边学:了解go中并发工作原理:挑战:利用并发方法更快地计算斐波纳契数

170. Go语言教程之边写边学:编写并测试程序 :概述网上银行项目

171. Go语言教程之边写边学:编写并测试程序 :开始编写测试

172. Go语言教程之边写边学:编写并测试程序 :编写银行核心程序包

173. Go语言教程之边写边学:编写并测试程序 :编写银行 API

174. Go语言教程之边写边学:编写并测试程序 :完成银行项目功能

175. Go语言教程之边写边学:了解如何在 Go 中处理错误

176. Go语言教程之边写边学:标准库:strings(1)

177. Go语言教程之边写边学:标准库:strings(2)

178. Go语言教程之边写边学:标准库:strings(3)

179. Go语言教程之边写边学:标准库:strings(4)

180. Go语言教程之边写边学:标准库:strings(5)

181. Go语言教程之边写边学:标准库:strings(6)