初级: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
文章来源于网络收集而来,版权归原创者所有,如有侵权请及时联系!
48. defer 函数的执行时机
对 defer 延迟执行的函数,会在调用它的函数结束时执行,而不是在调用它的语句块结束时执行,注意区分开。
比如在一个长时间执行的函数里,内部 for 循环中使用 defer 来清理每次迭代产生的资源调用,就会出现问题:
// 命令行参数指定目录名
// 遍历读取目录下的文件
func main() {
if len(os.Args) != 2 {
os.Exit(1)
}
dir := os.Args[1]
start, err := os.Stat(dir)
if err != nil || !start.IsDir() {
os.Exit(2)
}
var targets []string
filepath.Walk(dir, func(fPath string, fInfo os.FileInfo, err error) error {
if err != nil {
return err
}
if !fInfo.Mode().IsRegular() {
return nil
}
targets = append(targets, fPath)
return nil
})
for _, target := range targets {
f, err := os.Open(target)
if err != nil {
fmt.Println("bad target:", target, "error:", err) //error:too many open files
break
}
defer f.Close() // 在每次 for 语句块结束时,不会关闭文件资源
// 使用 f 资源
}
}
先创建 10000 个文件:
#!/bin/bash
for n in {1..10000}; do
echo content > "file${n}.txt"
done
运行效果:
解决办法:defer 延迟执行的函数写入匿名函数中:
// 目录遍历正常
func main() {
// ...
for _, target := range targets {
func() {
f, err := os.Open(target)
if err != nil {
fmt.Println("bad target:", target, "error:", err)
return // 在匿名函数内使用 return 代替 break 即可
}
defer f.Close() // 匿名函数执行结束,调用关闭文件资源
// 使用 f 资源
}()
}
}
当然你也可以去掉 defer,在文件资源使用完毕后,直接调用 f.Close()
来关闭。
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论