初级:1-34
- 1. 左大括号 { 不能单独放一行
- 2. 未使用的变量
- 3. 未使用的 import
- 4. 简短声明的变量只能在函数内部使用
- 5. 使用简短声明来重复声明变量
- 6. 不能使用简短声明来设置字段的值
- 7. 不小心覆盖了变量
- 8. 显式类型的变量无法使用 nil 来初始化
- 9. 直接使用值为 nil 的 slice、map
- 10. map 容量
- 11. string 类型的变量值不能为 nil
- 12. Array 类型的值作为函数参数
- 13. range 遍历 slice 和 array 时混淆了返回值
- 14. slice 和 array 其实是一维数据
- 15. 访问 map 中不存在的 key
- 16. string 类型的值是常量,不可更改
- 17. string 与 byte slice 之间的转换
- 18. string 与索引操作符
- 19. 字符串并不都是 UTF8 文本
- 20. 字符串的长度
- 21. 在多行 array、slice、map 语句中缺少 , 号
- 22. log.Fatal 和 log.Panic 不只是 log
- 23. 对内建数据结构的操作并不是同步的
- 24. range 迭代 string 得到的值
- 25. range 迭代 map
- 26. switch 中的 fallthrough 语句
- 27. 自增和自减运算
- 28. 按位取反
- 29. 运算符的优先级
- 30. 不导出的 struct 字段无法被 encode
- 31. 程序退出时还有 goroutine 在执行
- 32. 向无缓冲的 channel 发送数据,只要 receiver 准备好了就会立刻返回
- 33. 向已关闭的 channel 发送数据会造成 panic
- 34. 使用了值为 nil 的 channel
- 34. 若函数 receiver 传参是传值方式,则无法修改参数的原有值
中级:35-50
- 35. 关闭 HTTP 的响应体
- 36. 关闭 HTTP 连接
- 37. 将 JSON 中的数字解码为 interface 类型
- 38. struct、array、slice 和 map 的值比较
- 39. 从 panic 中恢复
- 40. 在 range 迭代 slice、array、map 时通过更新引用来更新元素
- 41. slice 中隐藏的数据
- 42. Slice 中数据的误用
- 43. 旧 slice
- 44. 类型声明与方法
- 45. 跳出 for-switch 和 for-select 代码块
- 46. for 语句中的迭代变量与闭包函数
- 47. defer 函数的参数值
- 48. defer 函数的执行时机
- 49. 失败的类型断言
- 50. 阻塞的 gorutinue 与资源泄露
高级:50-57
文章来源于网络收集而来,版权归原创者所有,如有侵权请及时联系!
36. 关闭 HTTP 连接
一些支持 HTTP1.1 或 HTTP1.0 配置了 connection: keep-alive
选项的服务器会保持一段时间的长连接。但标准库 "net/http" 的连接默认只在服务器主动要求关闭时才断开,所以你的程序可能会消耗完 socket 描述符。解决办法有 2 个,请求结束后:
- 直接设置请求变量的
Close
字段值为true
,每次请求结束后就会主动关闭连接。 - 设置 Header 请求头部选项
Connection: close
,然后服务器返回的响应头部也会有这个选项,此时 HTTP 标准库会主动断开连接。
// 主动关闭连接
func main() {
req, err := http.NewRequest("GET", "http://golang.org", nil)
checkError(err)
req.Close = true
//req.Header.Add("Connection", "close") // 等效的关闭方式
resp, err := http.DefaultClient.Do(req)
if resp != nil {
defer resp.Body.Close()
}
checkError(err)
body, err := ioutil.ReadAll(resp.Body)
checkError(err)
fmt.Println(string(body))
}
你可以创建一个自定义配置的 HTTP transport 客户端,用来取消 HTTP 全局的复用连接:
func main() {
tr := http.Transport{DisableKeepAlives: true}
client := http.Client{Transport: &tr}
resp, err := client.Get("https://golang.google.cn/")
if resp != nil {
defer resp.Body.Close()
}
checkError(err)
fmt.Println(resp.StatusCode) // 200
body, err := ioutil.ReadAll(resp.Body)
checkError(err)
fmt.Println(len(string(body)))
}
根据需求选择使用场景:
- 若你的程序要向同一服务器发大量请求,使用默认的保持长连接。
- 若你的程序要连接大量的服务器,且每台服务器只请求一两次,那收到请求后直接关闭连接。或增加最大文件打开数
fs.file-max
的值。
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论