(1)用外层有a,b 用 a,b,c:= 会创建所有变量,而不仅仅是新出现的c。
func fn() { var a, b int { a, b, c := 1, 1, 1 fmt.Println(a, b, c) } fmt.Println(a, b) }
(2)使用defer时 一定要用 xx()(err error)明示返回变量的名字,这样内层return err会进行一次赋值,不然内存retrun后,defer读的是外层err,是空的。
func fn() error { //初始化一个error var err error //注册defer处理 defer func() { //判断err(是外层err) if err != nil { fmt.Println("haha") } }() //某种场景提前返回 if 1 == 1 { //这种场景触发某种err,这边可能为了方便使用了 := 重建了局部err val, err := 1, errors.New("text string") _ = val //这边返回的是局部err,defer里无法捕获这个err return err } //返回全局err return err }
修改一下
func fn() (err error) { //注册defer处理 defer func() { //判断err(是外层err) if err != nil { fmt.Println("haha") } }() //某种场景提前返回 if 1 == 1 { //这种场景触发某种err,这边可能为了方便使用了 := 重建了局部err val, err := 1, errors.New("text string") _ = val //这边返回的是局部err,但是实际上还有个隐式的赋值,局部err赋值给全局err,然后defer就捕获的err就非nil了 return err } //返回全局err return }