易犯的五个Go编码错误

1. 使用 time.Parse()解析时间

导致获取的时间戳比真实的多 8 小时。即使设置了全局的time.Local也一样,更换成time.ParseInLocation即可,原因:

go 复制代码
 
func Parse(layout, value string) (Time, error) {
	// Optimize for RFC3339 as it accounts for over half of all representations.
	if layout == RFC3339 || layout == RFC3339Nano {
		if t, ok := parseRFC3339(value, Local); ok {
			return t, nil
		}
	}
    // 源码中写死的UTC时区
	return parse(layout, value, UTC, Local)
}

2. 不清楚 defer 参数是被立即计算的

go 复制代码
 
func Doing() error {
    var err error
    defer handlerErr(err) // handlerErr 永远得到的err是nil
    err = doSomething()
    return err
}

3. 循环中 defer

go 复制代码
 
func Doing() {
	mutex := sync.Mutex{}
	for i := 0; i < 1000000000; i++ {
		mutex.Lock()
		defer mutex.Unlock() // 死锁,因为要等方法结束后才Unlock
        // do something
	}
}

4. 未注意 net/http 中 ctx 生命周期

go 复制代码
 
package main
import (
    "context"
    "net/http"
)

func main() {
    http.HandleFunc("/createOrder", func(w http.ResponseWriter, r *http.Request) {
       ctx := r.Context()
       // 创建订单
       orderDo := createOrder(ctx, reqParams)

       // 异步推送订单到供应链
       go SubmitToSupplier(ctx, orderDo)
    })
    http.ListenAndServe(":8080", nil)
}

由于 ctx 在 socket 连接读取出错或 handleFunc 完成时会被 cancel,导致:

  1. 当浏览器或调用方的 clinet 主动关闭时,即使 createOrder 未运行完,ctx 也会收到 cancel 事件,如是里面有 DB 不在一个事务、或支持 ctx 的 IO 请求的操作,都会中断。
  2. 协程启动后,由于 handleFunc 运行结束,ctx 被 cancel,导致程序受协程方法运行受影响

5. 协程中未捕获 panic

导致主程序退出,建议统一封装启动协程方法,否则很容易忘记 recover panic

相关推荐
hayson41 分钟前
mcp-go 用法详解及源码解析
go·mcp
Joey_Chen3 小时前
【Golang开发】超详细!Golang开发环境搭建
后端·go
郭京京3 小时前
go日志包log
go
郭京京3 小时前
go函数被设计为一等公民
后端·go
江湖十年3 小时前
Go 项目中的 doc.go 文件是干嘛的?
后端·面试·go
HyggeBest21 小时前
Golang 并发原语 Sync Once
后端·go
zhuyasen2 天前
当Go框架拥有“大脑”,Sponge框架集成AI开发项目,从“手写”到一键“生成”业务逻辑代码
后端·go·ai编程
写代码的比利2 天前
Kratos 对接口进行加密转发处理的两个方法
go