Go赋值操作的关键细节

张开发
2026/4/10 16:48:17 15 分钟阅读

分享文章

Go赋值操作的关键细节
一、:短变量声明的细节1.作用域规则2.重复声明规则3类型推断二、 赋值操作的细节1.类型必须匹配2.多重赋值特性三、 零值初始化细节1.Go为所有类型提供零值四、指针赋值的细节1.基本指针操作理解这段代码的关键在于搞清楚两个概念普通变量、指针变量以及它们之间的关系。用“门牌号”和“房子”来理解变量x就像是一栋房子里面住着数字10。指针变量p就像是一张纸条上面写着这栋房子的门牌号内存地址。现在看代码的每一步var x int 10var p *int x1.建房子x盖好了里面放了10。2.写纸条p是张空白纸条x就是去查x的门牌号然后把门牌号写到p的纸条上。*p 203.改房子里的东西*p这个动作叫解引用。意思是拿起p这张纸条顺着上面的门牌号找到x这栋房子然后把里面的东西换成20。注意纸条没变还是那个门牌号但房子里的东西变了。fmt.Println(x) // 输出 204.结果x这栋房子里的内容已经是20了。因为p和x指向的是同一栋房子所以通过p改x自己也会变。为什么var q *int危险var q *int// *q 30 // ❌ 这里会崩溃q是一张空白纸条nil指针。纸上没有写任何门牌号。如果执行*q 30就相当于“拿着一张空白纸条去找房子强行往里面塞东西。”你根本不知道那个房子在哪儿内存地址无效操作系统为了保护其他程序的安全会直接终止程序Panic。2.结构体指针五、特殊赋值情况1.空白标识符 _2.常量赋值六、复合类型的赋值细节1.切片赋值2.映射赋值七、 函数相关的赋值细节1.命名返回值一旦起了名字这两个变量在函数一开始就被自动创建并初始化为零值result0,errnil。当你写一个裸return后面不带东西时Go 就默认“把当前result和err的值原样打包返回给调用方。”场景一b 0出错进入函数Go 自动创建result0,errnil。检测到b 0执行err errors.New(...)。执行裸return。此时result还是0err是新错误。返回(0, error)。场景二b ! 0正常进入函数Go 自动创建result0,errnil。跳过if执行result a / b。执行裸return。此时result是计算结果err还是nil。返回(计算结果, nil)。2.闭包中的变量捕获为什么错误写法里所有函数都输出3i不是每次循环都在变吗闭包捕获的是变量本身不是当时的值在错误写法中for循环里的i是同一个变量。每次循环只是修改了这个变量的值0 → 1 → 2 → 3 循环结束。闭包函数func(){ fmt.Println(i) }记住的是i这个变量本身门牌号而不是创建时的数值。等到你真正执行这些函数时比如f : funcs[0]; f()循环早就结束了i已经变成了3。i : i到底干了什么左边的i创建一个全新的、局部的变量作用域仅在本次循环内。右边的i是外层for循环的变量。效果把外层i当前的值复制给内层新的i。结果闭包捕获的是这个全新的内层变量它只属于这一次循环下次循环会再创建另一个新的。funcs : make([]func(), 0)相当于你买了一个空的 “待办事项清单” 本子。funcs append(funcs, func(){...})相当于往这个本子里 “记下一条新任务”。核心连接点类型func()这两行代码能配合工作的唯一桥梁就是括号里的类型func()。代码片段中文直译关键细节make([]func(), 0)制作一个切片这个切片只能用来存放func()类型的东西。指明了仓库的“货架规格”。func(){ fmt.Println(i) }这是一个匿名函数它的签名正是func()无参数、无返回值。生产了一个符合“货架规格”的货物。append(funcs, ...)把货物放到货架上。类型必须严格匹配否则编译报错。如果把它们拆成三个时间点来看逻辑会更清晰// 第一步声明与初始化建立空仓库 // 此时 funcs 的长度为 0容量为 0里面一个函数都没有。 funcs : make([]func(), 0) // 第二步循环制造与入库append 操作 for i : 0; i 3; i { // 此时生成了一个匿名函数对象货物 // append 函数检查这个货物是不是 func() 类型 ✅ 是的。 // 于是 append 将 funcs 扩容并把货物放在索引位置。 funcs append(funcs, func() { fmt.Println(i) }) } // 第三步此时 funcs 变成了一个有 3 个元素的切片 // 结构类似于[ 函数0 , 函数1 , 函数2 ]为什么要用接收你可能会注意到写的是funcs append(...)而不是单纯的append(...)。这是因为Go 的切片在扩容时底层数组的地址可能会变搬仓库。append函数会把新仓库的地址作为返回值返回。如果你不用funcs 接住它你就失去新仓库的钥匙了funcs还是指向那个旧的、没扩容的空仓库。避免的陷阱总结

更多文章