前言

哈喽,今天与大家聊一个比较冷门的高频面试题,关于切片的,Go语言中的切片原生支持并发吗?怎么样,心里有答案了嘛,带着你的思考我们一起来看一看这个知识点。本文转载自公众号「Golang梦工厂」,推荐关注!

实践检验真理

实践是检验真理的唯一标准,所以当我们遇到一个不确定的问题,直接写demo来验证,因为切片的特点,我们可以分多种情况来验证:

  1. 不指定索引,动态扩容并发向切片添加数据

func concurrentAppendSliceNotForceIndex() {sl := make([]int, 0)wg := sync.WaitGroup{}for index := 0; index < 100; index++{k := indexwg.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测试一下;

  1. 指定索引,指定容量并发向切片添加数据

func concurrentAppendSliceForceIndex() {sl := make([]int, 100)wg := sync.WaitGroup{}for index := 0; index < 100; index++{k := indexwg.Add(1)go func(num int) {sl[num] = numwg.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解决了该问题,感兴趣的可以看一下这个issue:https://github.com/kataras/iris/pull/1023#event-1777396646;

总结

针对上述问题,我们可以多种方法来解决切片并发安全的问题:

  1. 加互斥锁

  2. 使用channel串行化操作

  3. 使用sync.map代替切片

切片的问题还是比较容易解决,针对不同的场景可以选择不同的方案进行优化,你学会了吗?

Go 的切片支持并发吗?相关推荐

  1. Go语言切片原生支持并发吗?

    前言 哈喽,大家好,我是asong.今天与大家聊一个比较冷门的高频面试题,关于切片的,Go语言中的切片原生支持并发吗?怎么样,心里有答案了嘛,带着你的思考我们一起来看一看这个知识点. 实践检验真理 实 ...

  2. 003_支持并发的饿汉单例

    package com.zr.single;/*** 支持并发的饿汉单例*/ public class HungrySingletion {// 静态初始化器, 由JVM来保证线程安全.private ...

  3. 002_支持并发的内部类饿汉单例

    package com.zr.single;/*** 支持并发的内部类饿汉单例*/ public class HungryInnerSingletion {// 类级的静态的成员式内部类, 该内部类的 ...

  4. 001_支持并发的两次判空懒汉单例

    package com.zr.single;/*** 支持并发的两次判空懒汉单例*/ public class TwiceJudgeNullLazySingleton {private TwiceJu ...

  5. python实现文件共享_改进的一行Python实现文件共享--支持并发

    SimpleHTTPServer是python自带的一个简单的静态HTTP服务器, python2.x下只要切换到想要共享的目录简单地输入下面的命令,就可以打开该HTTP服务器: python -m ...

  6. 并发服务器的信号传递,使服务器支持并发、GIL全局解释器锁、死锁和Rlock、信号量、event事件、...

    服务器的并发实现: 服务端: importsocketfrom threading importThread"""服务端 1.要有固定的IP和PORT 2.24小时不间断 ...

  7. PHP双码率视频云转码服务系统源码 m3u8切片支持秒切及api上传和防盗功能,亲测极速播放

    网站里面存放视频在当前是一个刚需了,对于一些网站的视频存放途径可以是多种的.要么存放在oss云存储进行播放,要么直接存自己服务器.但是奈何服务器的带宽有限,因此可以进行视频切开分片加载.这样可以将视频 ...

  8. 对java支持并发的理解_Java并发知识(1)

    1. 进程和线程之间有什么不同? 一个进程是一个独立(self contained)的运行环境,它可以被看作一个程序或者一个应用.而线程是在进程中执行的一个任务.Java运行环境是一个包含了不同的类和 ...

  9. python多线程文件的数据续传_python实现支持并发、断点续传的Ftp程序

    一.要求 1.用户md5认证 2.支持多用户同时登陆(并发) 3.进入用户的命令行模式,支持cd切换目录,ls查看目录子文件 4.执行命令(ipconfig) 5.传输文件: a.支持断点续传 b.传 ...

最新文章

  1. MySQL查询区分大小写
  2. 亿级PV,常见性能优化策略总结与真实案例
  3. ZBrush关于遮罩的一些操作
  4. python __call__或者说func()()的理解
  5. HTML(XHTML)基础知识(三)——【image】
  6. [双十二优惠大放送] 我是小册姐,我因掘金小册而被创造,谢谢你们20天的陪伴...
  7. 爬取外网数据(twitter、facebook)-易数云可视化爬虫软件
  8. 计算机log是代表什么,Log是什么文件?Log文件可以删除吗?
  9. Proguard打包混淆报错:can't find superclass or interface
  10. 结构光三维重建(一)条纹结构光三维重建
  11. android商店账号密码错误,android app 自定义签名出现错误:Keystore was tampered with, or password was incorrect...
  12. 根据微信的code_url 给前端返回二维码图片--转换成base
  13. python csv文件和xlsx文件混杂时,提取指定列数据并合并
  14. c++软件开发面试旋极面试题_北京旋极信息技术股份有限公司2015招聘
  15. Jmeter接口压力测试--先登录后再测
  16. 高级计算机维修员职业资格证书,关于国家高级计算机维修职业资格证书在太原市怎么..._出版资格_帮考网...
  17. 判断一句话是否中文或者英文
  18. java毕业设计体育城场地预定系统前台源码+lw文档+mybatis+系统+mysql数据库+调试
  19. 【重要】2019年美赛(MCM/ICM)的参赛建议
  20. 帆软 FCRA 题库

热门文章

  1. RANSAC 算法拟合平面
  2. 使用SPSS进行一致性分析的相关步骤
  3. 外卖webapp1---express
  4. 2022广西最新建筑八大员(材料员)模拟考试试题及答案
  5. customErrors 小结
  6. 管理感悟:什么是对企业的忠诚(全新观点)
  7. 什么是LNA,与PA有什么区别
  8. 简单的J2EE(一)J2EE诞生的一些破事
  9. 一把王者的时间,我就学会了 Nginx
  10. 码农:曾diss我技术的人如今没工作找我,太具讽刺意义了!