Go语言协程
协程的特点
1.该任务的业务代码主动要求切换,即主动让出执行权限
2.发生了IO,导致执行阻塞(使用channel让协程阻塞)
与线程本质的不同
C#、java中我们执行多个线程,是通过时间片切换来进行的,要知道进行切换,程序需要保存上下文等信息,是比较消耗性能的
GO语言中的协程,没有上面这种切换,一定是通过协程主动放出权限,不是被动的。
例如:
C# 中创建两个线程

可以看到1和2是交替执行的

Go语言中用协程实现一下
runtime.GOMAXPROCS(1)


这个结果就是 执行了1 在执行2
上述两种方式来进行协程的切换
1.该任务的业务代码主动要求切换,即主动让出执行权限

2.通过channel进行阻塞

执行的结果一样

后面又继续执行1了
总结:http://www.infoq.com/cn/articles/knowledge-behind-goroutine
补充:2018/09/10 Go语言中的goroutine
一、Goroutine的并行
package main import (
"fmt"
"runtime"
"sync"
)
func main() {
runtime.GOMAXPROCS()
var wg sync.WaitGroup
wg.Add()
fmt.Println("Start Goroutines") go func() {
defer wg.Done()
for i:=;i<=;i++{
fmt.Print("")
}
}() go func() {
defer wg.Done()
for i:=;i<=;i++ {
fmt.Print("")
}
}() go func() {
defer wg.Done()
for i:=;i<=;i++ {
fmt.Print("")
}
}() fmt.Println("等待执行结束")
wg.Wait()
}
运行结果
Start Goroutines 等待执行结束
(1)runtime.GOMAXPROCS(1) 的作用是什么?
runtime包的GOMAXPROCS 函数。这个函数允许程序更改调度器可以使用的逻辑处理器的数量。如果不想在代码里做这个调用,也可以通过修改和这个函数名字一样的环境变量的值来更改逻辑处理器的数量。
(2)为什么先输出3后输出1和2?
如下有详细的讲解 https://studygolang.com/topics/5157?fr=sidebar ,不过经过各种实验发现这个执行顺序其实是变的,貌似纠结这个顺序也没有什么意思。
(3)WaitGroup 是什么意思?
WaitGroup 是一个计数信号量,可以用来记录并维护运行的goroutine。如果WaitGroup的值大于0,Wait 方法就会阻塞。为了减小WaitGroup 的值并最终释放main 函数,使用defer 声明在函数退出时
调用Done 方法。(defer 有点像C#当中的fianlly)
补充:调度算法中,如果多个goroutine中某个执行过长,此时会将其停止让给其他goroutine继续执行,待到其他都执行完成,在将其进行执行。如下图:G4和G5就进行了切换

如下代码可以验证上面的问题:
package main import (
"fmt"
"runtime"
"sync"
) var wg sync.WaitGroup func main() {
runtime.GOMAXPROCS() wg.Add() fmt.Println("Create Goroutines")
go printPrime("A")
go printPrime("B") fmt.Println("Waiting To Finish")
wg.Wait() fmt.Println("Terminating Program")
} func printPrime(prefix string) {
defer wg.Done() next:
for outer := ; outer < ; outer++ {
for inner := ; inner < outer; inner++ {
if outer%inner == {
continue next
}
}
fmt.Printf("%s:%d\n", prefix, outer)
}
fmt.Println("Completed", prefix)
}
printPrime 这个函数作用是查找显示 5000 以内的素数值,这是一个比较耗时的程序。
运行结果:数字比较多,不打印了,但是可以看到结果是A和B两个协程之间的切换。
上面的代码都是设置GOMAXPROCS为1的情况,给每个可用的核心分配一个逻辑处理器
runtime.GOMAXPROCS(runtime.NumCPU())
这样运行第一个程序的结果如下
Start Goroutines
等待执行结束133333333333333333333333333333333333333333333333333333333333333333333333333333333333333333333333333332111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222
可以看到1、2、3没有什么规律。
二、Goroutine的竞争
什么竞争状态:如果两个或者多个goroutine 在没有互相同步的情况下,访问某个共享的资源,并试图同时读和写这个资源,就处于相互竞争的状态,这种情况被称作竞争状态(race candition)
竞争状态是比较容易出现问题的地方,所以对一个共享资源的读和写操作必须是原子化的,换句话说,同一时刻只能有一个goroutine 对共享资源进行读和写操作。
package main import (
"fmt"
"runtime"
"sync"
) var (
counter int
wg sync.WaitGroup
) func main() {
wg.Add() go incCounter()
go incCounter() wg.Wait()
fmt.Println("Final Counter:", counter)
} func incCounter(id int) {
defer wg.Done()
for count := ; count < ; count++ {
value := counter
runtime.Gosched()
value++
counter = value
}
}
运行结果:
Final Counter:
变量 counter 会进行4 次读和写操作,每个goroutine 执行两次。但是,程序终止时,counter变量的值为2,这是因为两个协程之间产生了竞争,对同一个counter进行读写,下面这个图很好的诠释了为什么最后counter是2

补充:runtime.Gosched()用于让出CPU时间片。
三、使用Goroutine的锁
一种修正代码、消除竞争状态的办法是,使用Go 语言提供的锁机制,来锁住共享资源,
从而保证goroutine 的同步状态。Go 语言提供了传统的同步goroutine 的机制,就是对共享资源加锁。
1、原子函数
使用atomic包来提供对数值类型的安全访问。
package main import (
"fmt"
"runtime"
"sync"
"sync/atomic"
) var (
counter int64
wg sync.WaitGroup
) func main() {
wg.Add() go incCounter()
go incCounter() wg.Wait() fmt.Println("Final Counter:", counter)
} func incCounter(id int) {
defer wg.Done()
for count := ; count < ; count++ {
atomic.AddInt64(&counter, )
runtime.Gosched()
}
}
运行结果
Final Counter:
atmoic 包的AddInt64 函数。这个函数会同步整型值的加法,方法是强制同一时刻只能有一个goroutine 运行并完成这个加法操作。当goroutine 试图去调用任何原子函数时,这些goroutine 都会自动根据所引用的变量做同步处理。
另外两个有用的原子函数是LoadInt64 和StoreInt64。这两个函数提供了一种安全地读和写一个整型值的方式。
package main import (
"fmt"
"sync"
"sync/atomic"
"time"
) var (
shutdown int64
wg sync.WaitGroup
) func main() {
wg.Add() go doWork("A")
go doWork("B") time.Sleep( * time.Second) fmt.Println("Shutdown Now")
atomic.StoreInt64(&shutdown, )
wg.Wait()
} func doWork(name string) {
defer wg.Done() for {
fmt.Printf("Doing %s Work\n", name)
time.Sleep( * time.Millisecond) // Do we need to shutdown.
if atomic.LoadInt64(&shutdown) == {
fmt.Printf("Shutting %s Down\n", name)
break
}
}
}
主协程main函数执行了1秒,然后将shutdown中的值设置为1,A和B的协程中通过读取shutdown是否等于1来判断是否结束协程。
运行结果:
Doing A Work
Doing B Work
Doing A Work
Doing B Work
Doing A Work
Doing B Work
Doing A Work
Doing B Work
Shutdown Now
Shutting A Down
Shutting B Down
2、互斥锁
另一种同步访问共享资源的方式是使用互斥锁(mutex)。互斥锁这个名字来自互斥(mutualexclusion)的概念。互斥锁用于在代码上创建一个临界区,保证同一时间只有一个goroutine 可以执行这个临界区代码
package main import (
"fmt"
"runtime"
"sync"
) var (
counter int
wg sync.WaitGroup
mutex sync.Mutex
) func main() {
wg.Add() go incCounter()
go incCounter()
go incCounter() wg.Wait()
fmt.Printf("Final Counter: %d\n", counter)
} func incCounter(id int) {
defer wg.Done() for count := ; count < ; count++ {
mutex.Lock()
{
value := counter
runtime.Gosched()
value++
counter = value
}
mutex.Unlock()
}
}
运行结果:
Final Counter:
如上面的代码,最终的结果还是6.
mutex.lock he mutex.Unlock之间的代码,同一时刻只允许一个goroutine进入,所以保证数据的正确性。
补充对于 runtime.Gosched()的理解,有两个问题 1. 当执行了这句话,貌似让出了执行,那么它后面的语句还能执行么
Gosched可以参考下面的链接体会
https://nanxiao.me/go-runtime-gosched-introduction/
Go语言协程的更多相关文章
- 一个“蝇量级” C 语言协程库
协程(coroutine)顾名思义就是“协作的例程”(co-operative routines).跟具有操作系统概念的线程不一样,协程是在用户空间利用程序语言的语法语义就能实现逻辑上类似多任务的编程 ...
- go语言协程安全map
前言: 在go语言中 map 是很重要的数据结构.Map 是一种无序的键值对的集合.Map 最重要的一点是通过 key 来快速检索数据,key 类似于索引,指向数据的值.问题来了,这么安逸的 数据结构 ...
- Go语言协程并发---条件变量
package main import ( "fmt" "sync" "time" ) func main() { //要监听的变量 bit ...
- Go语言协程并发---读写锁sync.RWMutex
package main import ( "fmt" "sync" "time" ) /* 读写锁 多路只读 一路只写 读写互斥 */ / ...
- Go语言协程并发---互斥锁sync.Mutex
package main import ( "fmt" "sync" "time" ) /* mt.Lock() 抢锁 一次只能被一个协程锁 ...
- Go语言协程并发---等待组sync.WaitGroup
package main import ( "fmt" "sync" "time" ) /*等待组API介绍*/ func main071( ...
- Go语言协程并发---管道信号量应用
package main import ( "fmt" "math" "strconv" "time" ) /* ·10 ...
- Go语言协程并发---生产者消费者实例
package main import ( "fmt" "strconv" "time" ) /* 改进生产者消费者模型 ·生产者每秒生产一 ...
- Go语言协程并发---原子操作
package main import ( "fmt" "sync/atomic" ) /* 用原子来替换锁,其主要原因是: 原子操作由底层硬件支持,而锁则由操 ...
随机推荐
- Python常见的问题
1. new.init区别,如何实现单例模式,有什么优点 new是一个静态方法,init是一个实例方法new返回一个创建的实例,init什么都不返回new返回一个cls的实例时后面的init才能被调用 ...
- 2019年1月份A项目面试纪要
2019年1月份A项目面试纪要 本周二(1月22号),笔者接到了A项目的电话面试.这个面试来自A项目的客户,客户的后勤模块的几个顾问组成阵容强大的面试官团队.参加这个面试,让笔者感触良多,自己虽然在S ...
- 关于Keychain
1.Keychain 浅析 2.iOS的密码管理系统 Keychain的介绍和使用 3.iOS开发中,唯一标识的解决方案之keyChain+UUID
- DVWA 黑客攻防演练(九) SQL 盲注 SQL Injection (Blind)
上一篇文章谈及了 dvwa 中的SQL注入攻击,而这篇和上一篇内容很像,都是关于SQL注入攻击.和上一篇相比,上一篇的注入成功就马上得到所有用户的信息,这部分页面上不会返回一些很明显的信息供你调试,就 ...
- 网上都没有提到的教程:python捕获异常后,怎么输出错误文件和行号
1.假设输出不存在的变量 a try: print(a) except NameError as e: print('发生错误的文件:', e.__traceback__.tb_frame.f_glo ...
- 基于TCP 协议的RPC
前言: 环境: windown 10 Eclipse JDK 1.8 RPC的概念: RPC 是远程过程调用,是分布式网站的基础. 实验 SayHelloService.java 接口类,用于规范 S ...
- 【转载】FPGA算法设计随笔
FPGA设计算法依次需要完成MATLAB浮点仿真 MATLAB定点仿真 verilogHDL定点运算以及数据对比的流程.其中浮点到定点的转换尤为重要,需要在数据表示范围和精度之间做出权衡.另外掌握定点 ...
- SAP CRM Installed Bases(IBase)简介
SAP CRM使用Installed Base(以下简称IBase)来组织服务相关对象并进行管理.因为我在最近的工作中经常接触这个概念,所以学习了一点相关文档.下面是文档的翻译. 本文链接:https ...
- 【Python 08】汇率兑换2.0-1(字符串索引)
1.案例描述 设计一个汇率换算程序,其功能是将人民币转换为美元,或者美元转换为人民币. 增加功能:根据输入判断是人民币还是美元,进行相应的转换计算. 2.案例分析 3.字符串 两个双引号或单引号括起 ...
- Git拉取、提交、迁出、合并、删除分之命令
#拉取代码 git clone -b 分之名称 git地址 #提交代码 git add . //:注释,if是第一次提交: $ git add --all . (请注意后面有个英文点(表示是当前目录) ...