本文详解 go 并发编程中因复用同一底层数组切片导致 channel 接收端读取到重复或错误值的根本原因,并提供安全、高效的深拷贝实践方案。 本文详解 go 并发编程中因复用同一底层数组切片导致 channel 接收端读取到重复或错误值的根本原因,并提供安全、高效的深拷贝实践方案。在 Go 中,切片([]byte)并非值类型,而是包含三个字段的结构体:指向底层数组的指针、长度(len)和容量(cap)。这意味着当你将一个切片发送到 channel 时,实际传递的是该结构体的副本------但所有副本共享同一底层数组。若发送方 goroutine 在后续迭代中持续修改该数组(如 res[i]++),而接收方尚未消费前一次发送的切片,就会发生典型的数据竞争(data race):接收端看到的可能是已被覆盖的中间状态,从而出现值重复、跳变甚至 panic。你提供的代码正是这一问题的典型体现:func combinations(result chan []byte, length int, min byte, max byte) { res := make([]byte, length) // ← 底层数组在此分配 for i := 0; i < length; i++ { res[i] = min } result <- res // ← 发送切片(指针指向 res 底层数组) for { i := 0 for i = 0; i < length; i++ { if res[i] < max { res[i]++ // ← 直接修改底层数组! break } else { res[i] = 32 } } result <- res // ← 再次发送同一底层数组的切片 if i == length { close(result) return } }}尽管 fmt.Printf 前的打印显示"发送正确",但那只是发送瞬间的状态;接收端 range 循环读取时,res 的内容早已被下一轮循环改写------因此输出大量重复值(如 0100 出现两次),本质是竞态条件下的未定义行为。? 正确解法:每次发送前创建独立副本必须确保每个发送到 channel 的切片拥有专属底层数组。推荐使用 make + copy 进行浅层深拷贝(因元素为 byte,无需递归):func combinations(result chan []byte, length int, min byte, max byte) { res := make([]byte, length) for i := 0; i < length; i++ { res[i] = min } // 发送首组:先拷贝再发送 buf := make([]byte, length) copy(buf, res) result <- buf for { i := 0 for i = 0; i < length; i++ { if res[i] < max { res[i]++ break } else { res[i] = 32 } } // 关键修复:每次发送前都新建副本 buf := make([]byte, length) copy(buf, res) result <- buf if i == length { close(result) return } }}? 提示:也可封装为辅助函数提升可读性: 幻导航网 发现优质实用网站,开启网络探索之旅!
相关推荐
Dxy12393102161 小时前
Python 使用正则表达式将多个空格替换为一个空格qq_189807031 小时前
如何修改RAC数据库名_NID工具在集群环境下的改名步骤aXin_ya2 小时前
Redis 高级篇(最佳实践)zhangchaoxies2 小时前
如何检测SQL注入风险_利用模糊测试技术发现漏洞Luca_kill2 小时前
MCP数据采集革命:从传统爬虫到智能代理的技术进化zhangchaoxies2 小时前
CSS如何实现响应式弹性网格布局_配合media query修改flex-wrap属性霖霖总总2 小时前
[Redis小技巧32]Redis分布式锁的至暗时刻:从原理演进到时钟跳跃的终极博弈ZC跨境爬虫3 小时前
Scrapy分布式爬虫(单机模拟多节点):豆瓣Top250项目设置与数据流全解析sg_knight3 小时前
设计模式实战:命令模式(Command)