golang 学习过程中踩的坑

[他人总结]

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 包的时候就执行
执行顺序:

  1. 如果一个包的多个文件里面都有 init 函数, 按文件名排序之后的执行
  2. 同一个文件 import 了多个包, 则按照 import 的先后顺序执行
  3. 被递归 import 的包的初始化顺序与 import 顺序相反,例如:导入顺序 main –> A –> B –> C,则初始化顺序为 C –> B –> A –> main
  4. 一个包被其它多个包 import,但只能被初始化一次
  5. main 包总是被最后一个初始化
  6. 避免出现循环依赖, 例如: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 // 错误

检查除数是否可能为零

不确定的地方一定要做判断

:= 赋值声明语法糖使用不当可能踩坑

golang 赋值与声明语法糖使用注意事项

posted @ 2019-04-25 15:41  zlbeidou  阅读(349)  评论(0编辑  收藏  举报