美文网首页
Go条件变量

Go条件变量

作者: 骑蜗上高速 | 来源:发表于2019-12-27 15:02 被阅读0次

条件变量:
本身不是锁,但是经常和锁结合使用。对应有3个常用方法。wait,signal,broadcast
1、func (c *cond) Wait()
该函数的作用可归纳为三点:
a)阻塞等待条件变量满足
b)释放已经掌握的互斥锁,相当于cond.Unlock()。
注意:这两步为一个原子操作
c)当被唤醒,Wait()函数返回时,解除阻塞并重新获取互斥锁,相当于cond.lock()
2、func(c *cond) Signal()
单发通知,给一个正在等待(阻塞)在该条件变量上的goroutine发送通知。
3、func(c *cond) Broadcast()
广播通知,给正在等待(阻塞)在该条件变量上的所有goroutine发送通知。

流程:
1、创建条件变量:var cond sync.Cond。
2、指定条件变量用的锁:cond.L = new(sync.Mutex)。
3、cond.L.lock给公共区加锁。
4、判断是否到达阻塞条件(缓冲区满/空)---for 循环判断。
for len(ch) == cap(ch) {cond.wait()---1) 阻塞2)解锁3)加锁}。
5、访问公共区域 ----读、写数据。
6、解锁条件变量的锁cond.unlock()。
7、唤醒阻塞在条件变量上的对端。
代码示例:

package main

import (
    "fmt"
    "math/rand"
    "sync"
    "time"
)

var cond sync.Cond//定义全局条件变量

func producer3 (out chan <- int, idx int) {//定义生产者
        for {
            cond.L.Lock()//给公共缓冲区加锁
            for len(out) == 3 {//循环判断缓冲区容量,缓冲区满了,阻塞
                cond.Wait()//1)阻塞2)解锁3)加锁。 1和2是一个原子操作
            }
            num := rand.Intn(1000)//产生随机数
            out <- num //随机数写入管道
            fmt.Printf("生产者:%d,生产:%d\n", idx,num)
            cond.L.Unlock()//给公共缓冲区解锁
            cond.Signal()//唤醒被阻塞的消费者,进行消费数据
            time.Sleep(time.Millisecond*300)
        }
}
func consummer3(in <- chan int, idx int) {//定义消费者
    for {
        cond.L.Lock()//给公共缓冲区加锁
        for len(in) == 0 {//循环判断缓冲区容量,是否为空,为空阻塞,不为空,继续消费数据
            cond.Wait()//1)阻塞2)解锁3)加锁。1和2是一个原子操作
        }
            num :=  <- in   //从管道中消费随机数
        fmt.Printf("##########消费者:%d,消费:%d\n", idx,num)
        cond.L.Unlock()//给公共缓冲区解锁
        cond.Signal()//管道中数据被消费没了,唤醒被阻塞的生产者,进行生产数据
        time.Sleep(time.Millisecond * 200)
    }
}
func main() {
    ch := make(chan int,3)//定义管道,用于通信
    //quit := make(chan bool)//判断是否退出
    cond.L = new(sync.Mutex) //定义条件变量用的锁
    rand.Seed(time.Now().UnixNano())//随机数种子
    for i := 0;i < 5; i++ {
        go producer3(ch, i + 1)
    }
    for i := 0;i < 6; i++ {
        go consummer3(ch, i + 1)
    }
    //<- quit
    for {
        ;
    }
}

打印输出:

生产者:3,生产:702
##########消费者:6,消费:702
生产者:5,生产:386
生产者:4,生产:177
生产者:1,生产:953
##########消费者:3,消费:386
##########消费者:1,消费:177
##########消费者:2,消费:953
生产者:2,生产:201
##########消费者:4,消费:201
生产者:2,生产:902
生产者:5,生产:237
生产者:4,生产:642
##########消费者:4,消费:902
##########消费者:1,消费:237
##########消费者:3,消费:642

相关文章

  • Go条件变量

    条件变量:本身不是锁,但是经常和锁结合使用。对应有3个常用方法。wait,signal,broadcast1、fu...

  • Go语言学习笔记-基本程序结构-条件和循环

    循环 条件循环 无限循环 Go语言仅支持for循环 if条件 condition 必须为bool值 支持变量赋值 ...

  • go语言流程和函数(五)

    流程控制 if语句 go语言的if语句不需要括号,go语言的if还可以在条件判断的地方声明变量,go语言中的got...

  • GO语言初级学习之代码案例08(生产者消费者模型)

    GO语言初级学习之代码案例08-生产者消费者模型 @(go语言 黑马)[GO语言] 条件变量模拟生产者消费者模型 ...

  • go 安装配置(一)

    基本 安装gobrew install go 查看go环境变量go ev 设置go环境变量vim .bash_pr...

  • Go并发编程之美-条件变量

    一、前言 go语言类似Java JUC包也提供了一些列用于多线程之间进行同步的措施,比如低级的同步措施有 锁、CA...

  • Go之旅-变量

    Go语言变量有固定的数据类型,决定了变量内存的长度和存储格式。Go变量只能修改变量值,不能改变变量的数据类型。Go...

  • golang的helloworld

    go语言初识 go语言的变量

  • GO语言基础(六)

    定义变量 Go语言里面定义变量有很多种方式。使用var关键字是Go最基本的定义变量方式,与C语言不同的是Go把变量...

  • Go 并发实战 -- sync Cond

    前言 go中的sync.Cond也就是condition,是一个条件同步变量,与Java中Object的wait、...

网友评论

      本文标题:Go条件变量

      本文链接:https://www.haomeiwen.com/subject/epddoctx.html