Go新手最常踩的坑是环境、依赖和并发逻辑问题:未执行go mod init导致导入失败;goroutine往无缓冲chan发数据却不接收,造成永久阻塞泄漏。
Go语言新手在跑第一个项目时,最常掉进的坑不是语法写错,而是环境、依赖和并发逻辑这些“看不见的环节

import 路径全乱了很多新手克隆完项目,跳过 go mod init 直接 go run main.go,结果遇到 cannot find package "xxx" 或导入路径解析失败。这是因为 Go 1.11+ 默认启用 module 模式,所有 import 都基于模块路径,而非旧式 $GOPATH/src 结构。
go mod init your-module-name(模块名建议用 GitHub 地址,如 github.com/yourname/project)go.mod 但依赖缺失,运行 go mod tidy 自动下载并修正版本go.mod 里的 require 行——用 go get github.com/some/pkg@v1.2.3 更安全chan int 让协程永远卡住这是实战中最隐蔽的资源泄漏点。新手常写类似代码启动 goroutine 往无缓冲通道发数据,却忘了有谁来收——goroutine 就会永久阻塞,无法被调度器回收。
func badExample() {
ch := make(chan int)
go func() {
ch <- 1 // 永远卡在这儿
}()
// 没有 <-ch,也没 close(ch),goroutine 泄漏
}select + default 防阻塞ch := make(chan int, 1) 只能塞一次,再塞就阻塞context.WithTimeout 控制 goroutine 生命周期定义方法时选错接收者类型,是新手调试半天才发现的典型问题。比如想修改结构体字段,却用了值接收者,结果原对象纹丝不动。
type User struct{ Name string }
func (u User) SetName(n string) { u.Name = n } // ❌ 不生效
func (u *User) SetName(n string) { u.Name = n } // ✅ 生效*T 指针接收者真正卡住新手的,往往不是“不会写”,而是“不知道哪一步该做”。比如 go mod tidy 该在什么时候跑、chan 的缓冲要不要设、*T 接收者该不该加——这些细节不踩一遍坑,很难凭文档记住。