目录
- [他人总结]
- 首字母大写才是对外可见的
- 包的初始化函数顺序问题
- DB 连接泄漏问题
- err 常用写法
- goroutine 内的变量
- 指针可能是 nil
- 多层 map 未初始化
[他人总结]
https://chai2010.cn/advanced-go-programming-book/appendix/appendix-a-trap.html
http://km.oa.com/group/16106/articles/show/304264
首字母大写才是对外可见的
需要首字母大写才是对外可见的, 这是的对外指的是不同的 package, 特别需要注意的是 json 解析时的 struct 里面的字段名首字母必须大写, 否则无法解析, 同理还有 toml 配置解析等等
包的初始化函数顺序问题
包内的 init() 函数会在 import 包的时候就执行
执行顺序:
- 如果一个包的多个文件里面都有 init 函数, 按文件名排序之后的执行
- 同一个文件 import 了多个包, 则按照 import 的先后顺序执行
- 被递归 import 的包的初始化顺序与 import 顺序相反,例如:导入顺序 main –> A –> B –> C,则初始化顺序为 C –> B –> A –> main
- 一个包被其它多个包 import,但只能被初始化一次
- main 包总是被最后一个初始化
- 避免出现循环依赖, 例如:A –> B –> C –> A
DB 连接泄漏问题
sql.Open 和 DB.Close 并不能保证连接不泄漏, database/sql 其实是连接池, Open 之后并不马上进行连接, 只有实际查询的时候才发起连接.
Query 返回的 Rows 必须 Next 每一行数据才会断开连接, 之后把连接归还到连接池, 没有读完的数据比较手动 Rows.Close 才能保证连接不泄露.
err 常用写法
// 建议都这么写:
v, err := doSth()
if err != nil {
// 出错了
...
}
// 正常的逻辑
...
// 不建议下面这样写
// 原因是 **__err==nil__** 很容易被误写成 **__err!=nil__**:
if v, err := doSth(); err == nil {
// 正常的逻辑
...
} else {
// 出错了
...
}
goroutine 内的变量
// 错误
for i:=0; i < 100; i++ {
go func() {
fmt.Println(i)
}()
}
// 正确
for i:=0; i < 100; i++ {
go func(i int) {
fmt.Println(i)
}(i)
}
// 正确
for i:=0; i < 100; i++ {
i := i
go func() {
fmt.Println(i)
}()
}
指针可能是 nil
需要注意返回指针的函数调用者一定要判断指针是否为 nil
或者写带指针返回值的函数时加一个 error 或者 bool 出参, 这样调用方不得不判断是否逻辑异常, 可以一定程度避免调用方不检查的情况
多层 map 未初始化
对未初始化的 map 取值正常, 赋值出错
var m map[int]int
if m[5] == 5 { // 正确
...
}
m[5] = 5 // 错误
m := make(map[int]map[int]int)
m[5][5] = 5 // 错误
原文地址:https://www.cnblogs.com/zlbeidou/p/10768832.html
时间: 2024-11-06 09:37:46