当前位置: 首页 > news >正文

闭包,sync使用细节

代码

先看代码如下:

func main() {
    var a []int
    for i := 0; i < 100; i++ {
        go func() {
            a = append(a, i)
        }()
    }
    time.Sleep(2 * time.Second)
    fmt.Println(a)
}

这段测试代码是想要一个元素为0到100的切片,但是这一小段代码隐藏了很多的问题。

闭包函数

先看这段代码的执行结果:

[10 12 13 13 13 13 21 24 25 28 28 28 28 28 28 29 29 29 36 38 39 39 40 41 41 41 41 41 41 45 45 45 45 46 47 48 49 50 51 52 61 61 61 61 61 61 61 61 61 61 73 73 74 74 75 76 76 77 77 77 77 77 77 77 77 77 83 85 85 88 88 89 91 92 93 93 93 93 93 93 93 93 100 100 100 100 100 100 100]

可以发现有很多元素是相同的,这就是这段代码的第一个错误:使用闭包函数的时候,代码中这种传递参数i的方法并非深copy,而是传递变量指针。解释一下产生这种情况的原因:在并发执行时由于某一个协程修改了i的值,导致多个协程append的时候变量**i**的值发生变化,从而导致有多个重复的元素
将代码修改为:

func main() {
    var a []int
    for i := 0; i < 100; i++ {
        go func(i int) {
            a = append(a, i)
        }(i)
    }
    time.Sleep(2 * time.Second)
    fmt.Println(a)
}

执行结果为:

[5 4 8 7 2 12 15 13 14 24 22 23 25 18 21 17 20 28 29 31 30 32 33 34 35 36 37 38 39 41 40 42 44 50 45 48 49 55 51 52 53 54 46 47 57 56 58 59 60 65 61 62 63 64 68 66 67 70 69 72 74 71 73 75 76 80 77 78 79 86 81 82 83 85 89 87 88 84 90 91 92 95 93 94 97 96 98 99]

可以看到没有重复元素了,但是却缺少一些元素,这就引出了第二个问题。

多个协程的竞争问题

如上述代码,执行多次都会发现每次执行的结果都会少一些元素,其实真正的原因是没有对于竞争的协程加互斥锁,导致资源的丢失
解释这个问题要对go的数组、切片、以及append机制有一些了解,参考:

Arrays, slices (and strings): The mechanics of 'append'

现在知道我们声明的切片不同于数组,在每次append的时候我们会伴随着内存copy以达到自动扩容目的,在A协程读出a的内存数据时,B协程完成了写入操作,此时A继续append并赋值就会导致,协程B的更新结果丢失。
假如我们将切片换成数组就不存在这个问题:


func main() {
    var a [100]int
    for i := 0; i < 100; i++ {
        go func(i int) {
            // a = append(a, i)
            a[i] = i
        }(i)
    }
    time.Sleep(2 * time.Second)
    fmt.Println(a)
}

结果

[0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99]

或者互斥锁,

func main() {
    var a []int
    var mu sync.Mutex
    for i := 0; i < 100; i++ {
        go func(i int) {
            mu.Lock()
            a = append(a, i)
            mu.Unlock()
        }(i)
    }
    time.Sleep(2 * time.Second)
    fmt.Println(a)
}

结果

[1 0 2 9 7 8 10 4 5 3 11 6 12 14 13 16 15 23 20 21 22 19 25 24 17 26 18 27 28 29 32 30 31 34 35 36 40 33 37 39 38 42 43 41 44 51 45 49 50 55 52 53 48 54 46 47 57 56 58 59 60 64 61 62 63 68 72 70 71 74 69 75 73 65 66 67 76 79 77 78 85 80 81 82 83 84 86 88 87 90 89 91 92 93 96 94 95 97 98 99]

结论

  • 闭包使用注意变量传递是指针还是值,及注意闭包变量的两种传递方式。
  • 注意线程安全。

结语

希望大家一起学习,一起交流,一起进步!

联系我
qq:820932773
gmail: jdqaffairs@gmail.com

相关文章:

  • 力扣(LeetCode)21
  • uni-app项目数字滚动
  • 挂载磁盘报错“Structure needs cleaning”
  • js ES6 求数组的交集,并集,还有差集
  • 安装python包到指定虚拟环境
  • 验证码识别技术——15分钟带你突破各种复杂不定长验证码
  • 漫谈Java高并发方案
  • IOT物联网观察之金融科技新模式,可能改变世界的创新思维!
  • 数据分析之matplotlib.pyplot模块
  • C++基数排序
  • KindEditor 上传漏洞致近百个党政机关网站遭植入
  • MariaDB重置密码
  • 【ActiveMQ】- 发布/订阅模式
  • 效能改进之项目例会导入实践
  • iOS | NSProxy
  • 【许晓笛】 EOS 智能合约案例解析(3)
  • Git 使用集
  • Mac 鼠须管 Rime 输入法 安装五笔输入法 教程
  • Map集合、散列表、红黑树介绍
  • nginx(二):进阶配置介绍--rewrite用法,压缩,https虚拟主机等
  • Redux 中间件分析
  • Storybook 5.0正式发布:有史以来变化最大的版本\n
  • Sublime text 3 3103 注册码
  • 大快搜索数据爬虫技术实例安装教学篇
  • 经典排序算法及其 Java 实现
  • 开发基于以太坊智能合约的DApp
  • 快速体验 Sentinel 集群限流功能,只需简单几步
  • 前端知识点整理(待续)
  • 我是如何设计 Upload 上传组件的
  • 异步
  • 如何通过报表单元格右键控制报表跳转到不同链接地址 ...
  • ​iOS实时查看App运行日志
  • #include到底该写在哪
  • (Oracle)SQL优化技巧(一):分页查询
  • (Redis使用系列) Springboot 使用redis的List数据结构实现简单的排队功能场景 九
  • (八)五种元启发算法(DBO、LO、SWO、COA、LSO、KOA、GRO)求解无人机路径规划MATLAB
  • (黑客游戏)HackTheGame1.21 过关攻略
  • (四)docker:为mysql和java jar运行环境创建同一网络,容器互联
  • (推荐)叮当——中文语音对话机器人
  • (未解决)macOS matplotlib 中文是方框
  • (转)es进行聚合操作时提示Fielddata is disabled on text fields by default
  • (转)scrum常见工具列表
  • (总结)Linux下的暴力密码在线破解工具Hydra详解
  • (最简单,详细,直接上手)uniapp/vue中英文多语言切换
  • **PyTorch月学习计划 - 第一周;第6-7天: 自动梯度(Autograd)**
  • .bat批处理(三):变量声明、设置、拼接、截取
  • .Net Remoting(分离服务程序实现) - Part.3
  • .NET(C#、VB)APP开发——Smobiler平台控件介绍:Bluetooth组件
  • .NET/C# 利用 Walterlv.WeakEvents 高性能地定义和使用弱事件
  • .NET简谈互操作(五:基础知识之Dynamic平台调用)
  • /dev/sda2 is mounted; will not make a filesystem here!
  • @SuppressWarnings注解
  • [1127]图形打印 sdutOJ
  • [2]十道算法题【Java实现】
  • [2016.7 day.5] T2