16 Go并发编程(三): Go并发的传统同步机制

Go 传统同步机制

在《Go并发编程初探》中我们提到同步概念,所谓同步是相对异步而言,即串行相对于并行。
在学习Go通信机制时我们知道管道其实就是并发单元同步方式的一种,基于CSP并发模型,Go在语言原语上使管道作为核心设计,这是Go的设计哲学,也是Go所提倡的同步机制。然而,Go在标准包sync中也提供了传统的“共享内存式通信”的同步机制,对某些设计场景也需要这种同步方式,下面我们就来解析sync包提供的传统同步机制。

同步机制解决什么问题?

在并发编程中常常会遇到以下几种情况:

  • 在主协程中开辟的子协程依赖于主协程的生命周期,即“主死从随”,为了让子协程全部执行完成,主协程需要等待,但等待的时间是不定的,直接设置主协程睡眠略显粗暴,能否让子协程告诉父协程任务已完成?
  • 在并发编程中常常遇到多个协程共同操作公共资源的情况,如外部文件的并发读写,如果对这种资源的操作仍旧使用并行,则势必会造成混乱,能否对资源加锁,让各协程竞争锁串行操作?
  • 你也许也遇到过一些同一时间内只允许一个写入或同一时间内允许同时读取相同资源的情况,这种情况是典型的一读多写,对这种资源的操作该如何控制?
  • 当要求多个协程并发去执行一项任务,并只允许其中一个协程生效时,该如何处理?
  • 当一个协程正监听一个其他协程也可访问的资源,并等待该资源被其他协程修改,在该资源被修改时该协程从阻塞状态唤醒并继续执行,这种需求该如何解决?
  • 当一个资源被并发访问,且业务要求改资源必须在物理级别上并发安全,及在物理级别实现同一时间只被一个协程读写,Go有没有这种安全同步机制?

以上情况都是在并发编程中常见的资源安全问题,Go提供sync包实现安全的同步机制,以下我们一一解决上述遇到的问题:

1.等待组 sync.WaitGroup

Go同步包sync提供等待组,以解决主协程等待子协程完成任务的问题。

示例:

//sync同步:等待组 sync.WaitGroup
func BaseSync01() {

    wg := sync.WaitGroup{}

    wg.Add(1)
    go func() {
        for i := 1; i <= 10; i++ {
            fmt.Println("协程1走起!!!")
            time.Sleep(time.Second)
        }
        fmt.Println("=====协程1搞定=====!!!")
        wg.Done()
    }()

    wg.Add(1)
    go func() {
        for i := 1; i <= 7; i++ {
            fmt.Println("协程2走起!!!")
            time.Sleep(time.Second)
        }
        fmt.Println("=====协程2搞定=====!!!")
        wg.Done()
    }()

    wg.Add(1)
    go func() {
        for i := 1; i <= 5; i++ {
            fmt.Println("协程3走起!!!")
            time.Sleep(time.Second)
        }
        fmt.Println("=====协程3搞定=====!!!")
        wg.Done()
    }()

    for i := 1; i <= 3; i++ {
        fmt.Println("主协程走起!!!")
        time.Sleep(time.Second)
    }

    wg.Wait()
    fmt.Println("=====全部搞定=====!!!")

}
2.同步锁/互斥锁

所谓互斥锁,synv.Mutex ,保证被锁定资源不被其他协程占用,即被加锁的对象在同一时间只允许一个协程读或写。

示例:

func BaseSync02() {
    //申请一个锁
    mutex := sync.Mutex{}

    myWalet := 200

    //开20个协程
    for i := 1; i <= 100; i++ {
        go func(n int) {
            //没个协程分100次给我的钱包发1元
            for j := 1; j <= 1000; j++ {
                mutex.Lock()
                myWalet += 1
                mutex.Unlock()
                //fmt.Printf("协程%d,第%d次给我发1元红包\n",n,j)
            }
        }(i)
    }

    time.Sleep(time.Second * 3)
    fmt.Println("我的钱包现在为:", myWalet)
}
3.读写锁

所谓读协程snyc.RWMutex,实现业务中对资源一写多读的情况 ,如对数据库读写,为保证数据原子性,同一时间只允许一个协程写资源,禁止其他写入或读取,或同一时间允许多个协程读取资源,但禁止任何协程写入。

示例:

func BaseSync03() {
    rwm := sync.RWMutex{}
    myWallet := 200

    //开启3个写协程
    for i := 1; i <= 3; i++ {
        go func(n int) {
            for j := 1; j <= 100; j++ {
                rwm.Lock()
                myWallet += 1
                fmt.Println("写协程", i, "抢到写锁,修改金额为:", myWallet)
                rwm.Unlock()
                time.Sleep(time.Microsecond * 200)
            }
        }(i)
    }

    //开启1000个读协程
    for i := 1; i <= 1000; i++ {
        go func(n int) {
            runtime.Gosched()
            rwm.RLock()
            fmt.Println("读协程", n, "读到钱包金额:", myWallet)
            rwm.RUnlock()
        }(i)
    }

    time.Sleep(time.Second * 10)
    fmt.Println("最后读取金额为:", myWallet)

}
4.只执行一次 sync.Once

多协程调用中对一个任务只允许执行一次

示例:

//案例:杀死比尔,可以开多个协程去杀,但人只能死一次
type People struct {
    Name  string
    Alive bool
}

func Kill(p *People) {
    p.Alive = false
    fmt.Println("Bill:我被杀了...")
}

func BaseSync04() {
    once := sync.Once{}
    bill := People{"Bill", true}
    wg := sync.WaitGroup{}

    //开启三个协程去杀比尔
    for i := 1; i <= 10; i++ {
        wg.Add(1)
        go func() {
            fmt.Println("去杀Bill!!!")
            once.Do(func() {
                Kill(&bill)
            })

            wg.Done()
        }()
    }

    wg.Wait()
    fmt.Println("杀死比尔任务完成!!!")

}
5.条件变量 sync.Cond

示例:

/*
场景:
    1.监听比特币涨跌
    2.比特币涨,走投资协程,
    3.比特币跌,马上停止投资
    4.各协程监听比特币价格
*/
func BaseSync05() {
    //申请一个条件变量
    cond := sync.NewCond(&sync.Mutex{})

    //被监听的变量
    bitCoinRaising := false

    //涨协程变量修改并广播
    go func() {
        for {
            time.Sleep(time.Second * 3)
            cond.L.Lock()
            bitCoinRaising = true
            cond.Broadcast()
            cond.L.Unlock()
        }
    }()

    //跌协程变量修改并广播
    go func() {
        ticker := time.NewTicker(time.Second * 5)
        for {
            <-ticker.C
            cond.L.Lock()
            bitCoinRaising = false
            cond.Broadcast()
            cond.L.Unlock()
        }

    }()

    //监听条件变量的主协程阻塞等待
    for {
        cond.L.Lock()
        //不断循环监听变量
        if !bitCoinRaising {
            fmt.Println("比特币没涨,先暂停下")
            cond.Wait()
            fmt.Println("比特币涨了,快点买买买,发财了发财了!!!")
        }
        cond.L.Unlock()
    }

}
6.原子操作 sync.atomic

物理级别实现资源读写的原子性,从根本上杜绝并发不安全的问题,但其主要缺陷是只支持对基本数据类型的操作,对其他类型则无能为力。

示例:

func BaseSync06() {
    var myWalet int64
    myWalet = 200

    //开20个协程
    for i := 1; i <= 100; i++ {
        go func(n int) {
            //没个协程分100次给我的钱包发1元
            for j := 1; j <= 1000; j++ {
                atomic.AddInt64(&myWalet, 1)
                //fmt.Printf("协程%d,第%d次给我发1元红包\n",n,j)
            }

        }(i)
    }

    time.Sleep(time.Second * 3)
    fmt.Println("我的钱包现在为:", myWalet)
}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,529评论 5 475
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,015评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,409评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,385评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,387评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,466评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,880评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,528评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,727评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,528评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,602评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,302评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,873评论 3 306
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,890评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,132评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,777评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,310评论 2 342

推荐阅读更多精彩内容