代码

先看代码如下:

func main() {var a []intfor 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 []intfor 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]intfor 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 []intvar mu sync.Mutexfor 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

闭包,sync使用细节相关推荐

  1. let和闭包有啥关系

    前言 闭包这个概念并不是 JavaScript 的专利,本篇中描述的闭包均是 JavaScript 中的闭包,其相关的描述也均是围绕 JavaScript 来的.在 JavaScript 中闭包是很常 ...

  2. JavaScript内部原理实践——真的懂JavaScript吗?(转)

    通过翻译了Dmitry A.Soshnikov的关于ECMAScript-262-3 JavaScript内部原理的文章, 从理论角度对JavaScript中部分特性的内部工作机制有了一定的了解. 但 ...

  3. Scala学习笔记-详细记录学习中遇到的知识点

    目录 输入 print输出 变量 数据类型 标识符与关键字 运算符优先级 if else for循环 yield生成器 break与continue 函数 局部函数 惰性lazy 异常 类与对象 构造 ...

  4. 韩顺平 javascript教学视频_学习笔记20_多态经典案例_补讲闭包细节

    多态经典案例 经典案例,看代码: <html> <head> <script language="javascript"> function M ...

  5. Android 天气APP(九)细节优化、必应每日一图

    上一篇:Android 天气APP(八)城市切换 之 自定义弹窗与使用 重新定位.必应每日一图 新版------------------- 一.封装定位 二.重新定位 三.必应每日一图 ① 添加必应接 ...

  6. 如何用C语言改变宏定义的大小,C语言中宏定义使用的小细节

    C语言中宏定义使用的小细节 #pragma#pragma 预处理指令详解 在所有的预处理指令中,#Pragma 指令可能是最复杂的了,它的作用是设定编译器的状态或者是指示编译器完成一些特定的动作.#p ...

  7. swift3.0之闭包

    Swift 相比原先的 Objective-C 最重要的优点之一,就是对函数式编程提供了更好的支持. Swift 提供了更多的语法和一些新特性来增强函数式编程的能力,本文就在这方面进行一些讨论. Sw ...

  8. C#和Java的闭包-Jon谈《The Beauty of Closures》

    第一段略... 大多数讲闭包的文章都是说函数式语言,因为它们往往对闭包的支持最完善.当你在使用函数式语言时,很可能已经清楚了解了什么是闭包,所以我想写一篇在经典OO语言出现的闭包有什么用处应该也是很合 ...

  9. 如何设计一门语言(七)——闭包、lambda和interface

    人们都很喜欢讨论闭包这个概念.其实这个概念对于写代码来讲一点用都没有,写代码只需要掌握好lambda表达式和class+interface的语义就行了.基本上只有在写编译器和虚拟机的时候才需要管什么是 ...

最新文章

  1. cocos2d 从v1.x升级到v2.x需要注意的几个地方
  2. MySql的安装、配置(转)
  3. uboot主循环main_loop
  4. aws rds监控慢sql_探索AWS RDS SQL Server上SQL Server集成服务(SSIS)
  5. 从0开始搭建基于Zookeeper的Spark集群
  6. 使用OpenBTS基站测试物联网模块安全性
  7. 大数据入门第零天——总体课程体系概述
  8. iOS 非越狱下的代码注入
  9. 转:飝兒物語的“Linux创建、删除文件夹”
  10. Linux之《荒岛余生》(三)内存篇
  11. 兄弟俩今年的年龄和是35岁,当哥哥像弟弟现在这样大时,弟弟的年龄恰好是哥哥年龄的一半,哥哥今年几岁?
  12. python获得list的维度
  13. R树:处理空间存储问题
  14. XML 的 XPath 语法
  15. nginx反向代理地址是域名时请求不通的情况
  16. vue el-table 动态添加行、删除行
  17. Jackson转换JSON字符串
  18. 发光二极管封装常见方法有哪些?温升效应又是什么?
  19. 华丽家族股东大会21项议案全被否
  20. Android输入法挤乱布局问题

热门文章

  1. python爬虫豆瓣250_python爬虫二 爬取豆瓣Top250上
  2. AspectJ基于xml和基于注解
  3. PHP常用字符串函数小结
  4. MySQL源码包编译安装
  5. 从头学Qt Quick(3)-- 用QML写一个简单的颜色选择器
  6. 博弈论入门 HDU 1850
  7. DropDownList 控件不能触发SelectedIndexChanged 事件的另一个原因
  8. jps: command not found
  9. linux下生成源程序控制流图,Linux下控制(统计)文件的生成的C代码实现
  10. 计算机网络第5章(传输层)