Go语言切片原生支持并发吗?
2023-02-25 18:07:03 时间
实践检验真理
实践是检验真理的唯一标准,所以当我们遇到一个不确定的问题,直接写demo来验证,因为切片的特点,我们可以分多种情况来验证:
- 不指定索引,动态扩容并发向切片添加数据。
func concurrentAppendSliceNotForceIndex() {
sl := make([]int, 0)
wg := sync.WaitGroup{}
for index := 0; index < 100; index++{
k := index
wg.Add(1)
go func(num int) {
sl = append(sl, num)
wg.Done()
}(k)
}
wg.Wait()
fmt.Printf("final len(sl)=%d cap(sl)=%d\n", len(sl), cap(sl))
}
通过打印数据发现每次的结果都不一致,先不急出结论,我们在写其他的demo测试一下:
- 指定索引,指定容量并发向切片添加数据。
func concurrentAppendSliceForceIndex() {
sl := make([]int, 100)
wg := sync.WaitGroup{}
for index := 0; index < 100; index++{
k := index
wg.Add(1)
go func(num int) {
sl[num] = num
wg.Done()
}(k)
}
wg.Wait()
fmt.Printf("final len(sl)=%d cap(sl)=%d\n", len(sl), cap(sl))
}
通过结果我们可以发现符合我们的预期,长度和容量都是100,所以说slice支持并发吗?
slice支持并发吗?
我们都知道切片是对数组的抽象,其底层就是数组,在并发下写数据到相同的索引位会被覆盖,并且切片也有自动扩容的功能,当切片要进行扩容时,就要替换底层的数组,在切换底层数组时,多个goroutine是同时运行的,哪个goroutine先运行是不确定的,不论哪个goroutine先写入内存,肯定就有一次写入会覆盖之前的写入,所以在动态扩容时并发写入数组是不安全的;
所以当别人问你slice支持并发时,你就可以这样回答它:
当指定索引使用切片时,切片是支持并发读写索引区的数据的,但是索引区的数据在并发时会被覆盖的;当不指定索引切片时,并且切片动态扩容时,并发场景下扩容会被覆盖,所以切片是不支持并发的~。
github上著名的iris框架也曾遇到过切片动态扩容导致webscoket连接数减少的bug,最终采用sync.map解决了该问题。
总结
针对上述问题,我们可以多种方法来解决切片并发安全的问题:
- 加互斥锁
- 使用channel串行化操作
- 使用sync.map代替切片
切片的问题还是比较容易解决,针对不同的场景可以选择不同的方案进行优化,你学会了吗?
相关文章
- bash 脚本中关于变量的一些用法总结
- 深入理解Linux系统调用
- 腾讯灯塔融合引擎的设计与实践
- 纺织供应链中的金融大数据风控体系
- 数据集成平台 - SeaTunnel V2 架构演进
- 如何在OpenHarmony上进行双网卡设置
- 增强分析将是2023年的大趋势
- 如何让OpenHarmony编译速度“狂飙”
- Windows 10 / Windows 11 学院:如何禁用微软新版 Edge 浏览器的圆角设计
- 长见识,让大家看看什么是垃圾代码
- Linux 过去 3 年大幅优化 AMD Ryzen 处理器,综合性能提升 15%
- Mozilla 希望支持 Windows 7 / Windows 8.1 延长至 2024 年,扩大火狐浏览器 Firefox 的市场份额
- Linux 系统下,如下文件名中含有空格,该如何处理?
- 五个适合视力障碍者的 Linux 发行版
- 微软 Edge 浏览器将改用 Adobe Acrobat 的 PDF 渲染引擎
- 七个优秀的基于 Gentoo Linux 的发行版
- Fedora Budgie / Sway 两个 Spins 版本截图曝光
- 安装 Linux 后绝对不要做的八件事
- 因为用了C语言,Linux内核代码一团糟
- OpenHarmony富设备移植指南(四)第三方内核适配与定制