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" ) /* 用原子来替换锁,其主要原因是: 原子操作由底层硬件支持,而锁则由操 ...
随机推荐
- [JS设计模式]:工厂模式(3)
简单工厂模式是由一个方法来决定到底要创建哪个类的实例, 而这些实例经常都拥有相同的接口. 这种模式主要用在所实例化的类型在编译期并不能确定, 而是在执行期决定的情况. 说的通俗点,就像公司茶水间的饮料 ...
- 浏览器登录Dynamics 365 CE没毛病,程序连接却报错。
摘要: 微软动态CRM专家罗勇 ,回复308或者20190308可方便获取本文,同时可以在第一间得到我发布的最新博文信息,follow me!我的网站是 www.luoyong.me . 今天我做实验 ...
- 案例解析|政府信息化的BI建设应用 .
一.行业背景 某建设厅综合监管信息化平台,是政企业务协同的平台之一,同时兼具协作.门户.办公应用集成.用户权限管理等多项功能.在此要求基础上,选择中间件基础技术平台,可以在最大程度满足平台功能需求的前 ...
- Servlet工作原理解析 《深入分析java web 技术内幕》第九章
参考关于servblet的相关文章 侧重概况:https://blog.csdn.net/levycc/article/details/50728921 ibm的相关:https://www.ibm. ...
- python 爬取全本免费小说网的小说
这几天朋友说想看电子书,但是只能在网上看,不能下载到本地后看,问我有啥办法?我找了好几个小说网址看了下,你只能直接在网上看,要下载txt要冲钱买会员,而且还不能在浏览器上直接复制粘贴.之后我就想到py ...
- Windows系统XAMPP安装Moodle教程
一.安装工具下载: 系统环境: Operating System: Windows 10 Enterprise 64-bit (10.0, Build 17134) 集成软件: XAMPP Versi ...
- ASP.NET Zero--开发指南
ASP.NET Zero--开发指南(Lyhcee 译) 01. 前期介绍 02. 前期要求 03. 解决方案结构(层) 04. 前端应用程序 05. 后端应用程序 06.WEB.HOST应用程序 0 ...
- QQ邮箱开启SMTP方法如何授权
步骤一: 在打开的邮箱中心,进入设置 步骤二 从邮箱设置中心,进入帐户 步骤三 在这里可以看到POP3/SMTP服务被关闭了,因此客户端会收不到邮件,我们来开启它,查看该服务为已开启时,就可以用客户端 ...
- Clickhouse v18编译记录
简介 ClickHouse是"战斗民族"俄罗斯搜索巨头Yandex公司开源的一个极具"战斗力"的实时数据分析数据库,是面向 OLAP 的分布式列式DBMS,圈内 ...
- LeetCode算法题-Two Sum IV - Input is a BST(Java实现)
这是悦乐书的第280次更新,第296篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第148题(顺位题号是653).给定二进制搜索树和目标数,如果BST中存在两个元素,使得 ...