golang 并发 append

go中,切片可以算是我们最常用的结构之一,但是如果不注意的话,在并发情况下,对同一个切片进行append,极有可能会造成线程不安全的情况。

非线程安全现象

如,以下例子:

func TestOther(t *testing.T) {
	testMap := []int{1,2,3,4}

	wg := sync.WaitGroup{}
	wg.Add(4)
	
	m := make([]int, 0)
	
	for _, v := range testMap {
		go func(v int) {
			defer wg.Done()

			m = append(m, v)
		}(v)
	}
	wg.Wait()

	t.Log(m)
}

以上代码,不同次运行时,输出以下这类不符合预期的结果:

这就是因为线程不安全导致的 (实际业务中,可以通过 go run -race main.go 进行检测程序的安全性)

原因分析

slice的数据结构:

type slice struct {
    array unsafe.Pointer
    len   int
    cap   int
}
1.2 slice - 图1
slice结构示意图

使用append向Slice追加元素时,如果Slice空间不足,将会触发Slice扩容,扩容实际上重新一配一块更大的内存,将原Slice数据拷贝进新Slice,然后返回新Slice,扩容后再将数据追加进去。

在并发情况下,如果该slice始终空间不足,那么其是线程安全的,因为每次append实际都是新生成的内存,不存在抢占的情况。但是,当slice空间充足,也即是cap>len, 有剩余的空间时,比如说,下一个空闲内存是a, 那么并发情况下,就会出现多个线程抢占往a中写数据的情况。

slice扩容遵从以下原则:
如果原Slice容量小于1024,则新Slice容量将扩大为原来的2倍;
如果原Slice容量大于等于1024,则新Slice容量将扩大为原来的1.25倍;

所以,根据以上的原则,在程序运行后,是很容易就会出现有空闲空间的情况,也就会造成线程不安全的产生。

解决办法

针对内存占用,我们最直接简单的办法就是给内存加锁,如下:

func TestOther(t *testing.T) {
	testMap := []int{1,2,3,4}

	wg := sync.WaitGroup{}
	wg.Add(4)
	
	m := make([]int, 0)
	
	var lock sync.Mutex
	
	for _, v := range testMap {
		go func(v int) {
			defer wg.Done()

			lock.Lock()
			m = append(m, v)
			lock.Unlock()
			
		}(v)
	}
	wg.Wait()

	t.Log(m)
}

运行结果:

通过内存加锁,确保,同一时刻,只有一个线程对该块内存进行append操作,这就从根源上避免了抢占的问题。

精选应用推荐

安全便捷的密码管理工具
安全加密 跨设备同步 一键填充 免费使用
云幻梦密码本 - 安全密码管理
★★★★★
4.8 (12万+评价)
云幻梦密码本是一款专业级密码管理工具,采用银行级加密技术保护您的所有密码和个人信息。支持跨设备同步、一键自动填充、安全密码生成和生物识别登录等功能,让您的数字生活更加安全便捷。
应用截图
📦 68.5 MB
🔄 版本 2.5.1
📱 Android 8.0+
500万+下载

资源搜索推荐

一站式资源搜索平台
资源搜索 多源聚合 免费使用 无广告
千搜123 | 资源搜索
🔍
智能搜索
📁
多类资源
快速响应
🛡️
安全访问
千搜123是一个强大的资源搜索网站,聚合了多种资源搜索引擎,提供文档、软件、影视、音乐、学习资料等多种资源的快速搜索服务。界面简洁无广告,搜索结果精准,是您寻找各类资源的得力助手。
  • 聚合多个优质资源搜索引擎,一站式搜索
  • 支持文档、软件、影视、学习资料等多种资源类型
  • 界面简洁,无干扰广告,专注搜索体验
  • 搜索结果快速准确,节省您的时间
🌐 网站访问
无需下载
📅 持续更新
👨‍💻 免费服务

0 评论
最新
最旧 最多投票
内联反馈
查看所有评论
滚动至顶部