Go 语言学习笔记-select、锁和条件变量

select

  • select 的作用:

    通过 select 可以监听 channel 上的数据流动。

  • select 的用法:

    与 switch case 语法类似。但是 case 后面必须是 IO 操作,不可以人一些判断表达式。

    select {
    case <-chan1:
      // 如果 chan1 成功读到数据,则进行该 case 处理语句
    case chan2 <- 1:
      //如果成功向 chan2 写入数据,则进行该 case 处理语句
    default:
      // 如果上面都没有成功,则进入 default 处理流程
    }
    
  • 注意:

    • 监听的 case 中,没有满足监听条件,阻塞。
    • 监听的 case 中,有多个满足监听条件,任选一个执行。
    • 可以使用 default 来处理所有case 都不满足监听条件的状况。通常不用(会产生忙轮询)。
    • select 自身不带有循环机制,需借助外层 for 来循环监听。
    • break 跳出 select 中的 case 选项。类似于 switch 中的用法。

超时处理

有时候会出现 goroutine 阻塞的情况,可以使用 select 来设置超时,通过如下的方式实现:

func main() {
  c := make(chan int)
  o := make(chan bool)
  go func() {
    for {
      select {
      case v := <-c:
        fmt.Println(v)
      case <time.After(5 * time.Second):
        fmt.Println("timeout"):
        o <- true
        break
      }
    }
  }()
  // c <-66  // 注释掉,引发 timeout
  <-o
}

锁和条件变量

什么是锁

就是某个协程(线程)在访问某个资源时先锁住,防止其他协程的访问,等访问完毕解锁后其他协程再来加锁进行访问。

死锁

死锁不是锁的一种,是一种错误使用锁导致的现象。

死锁是指两个或两个以上的进程在进行过程中,由于竞争资源或者由于彼此通信而造成的一种阻塞的现象,若无外力作用,他们都将无法推进下去。此时称系统处于死锁状态或系统产生了死锁。

  • 常见的场景:
    • 单 go 程自己死锁
package main

import "fmt"

func main() {
  ch := make(chan int)
  ch <- 1
  fmt.Println("send")
  go func() {
    <- ch
    fmt.Println("received")
  }()
  fmt.Println("over")
}

互斥锁

每个资源都对应一个可称为 “互斥锁” 的标记,这个标记用来保证在任意时刻,只能有一个协程(线程)访问该资源。其它的协程只能等待。

互斥锁是传统并发编程对共享资源进行访问控制的主要手段。它由标准库 sync 中的 Mutex 结构体类型表示。 sync.Mutex 类型只有两个公开的指针方法,Lock 和 Unlock。Lock 锁定当前的共享资源,Unlock 进行解锁。

在使用互斥锁时,一定要注意:对资源操作完成后,一定要解锁,否则会出现流程执行异常、死锁等问题。通常借助 defer。锁定后,立即使用 defer 语句保证互斥锁及时解锁。如下所示:

var mutex sync.Mutex

func write() {
  mutex.Lock()
  defer mutex.Unlock()
}

读写锁

互斥锁的本质是当一个 goroutine 访问的时候,其他 goroutine 都不能访问。这样在资源同步,避免竞争的同时也降低了程序的并发性能。程序由原来的并行执行变成了串行执行。

读写锁可以让多个读操作并发,同时读取,但是对于写操作是完全互斥的。也就是说,当一个 goroutine 进行写操作的时候,其他 goroutine 既不能进行读操作,也不能进行写操作。

读时共享,写时独占。写锁优先级比读锁高。

  • “写锁定”和“读锁定”

    func (*RWMutex) Lock()
    func (*RWMutex) Unlock()
    
  • “读锁定”和“写锁定”

    func (*RWMutex) RLock()
    func (*RWMutex) RUnlock()
    

条件变量

条件变量的作用并不能保证在同一时刻仅有一个协程(线程)访问某个共享的数据资源,而是在对应的共享数据的状态发生变化时,通知阻塞在某个条件上的协程(线程)。条件变量不是锁,在并发中不能达到同步的目的,因此条件变量总是与锁一块使用

Go 标准库中的 sys.Cond 类型代表了条件变量。条件变量要与锁(互斥锁或者读写锁)一起使用。成员变量 L 代表与条件变量搭配使用的锁。

type Cond struct {
  noCopy noCopy
  L Locker
  notify notifyList
  checker copyChecker
}
  • 对应的 3 个常用方法,Wait、Signal、Broadcast:
    • func (c *Cond) Wait()
      该函数的作用:
      • 阻塞等待条件变量满足
      • 释放已掌握的互斥锁,相当于 cond.Unlock()。注意:两步为原子操作(不会被线程调度机制打断的操作;这种操作一旦开始,就一直运行到结束)
      • 当被唤醒,Wait() 函数返回时,解除阻塞并重新获取互斥锁。相当于 cond.Lock()
    • func (c *Cond) Signal()
      单发通知,给一个正等待(阻塞)在该条件变量上的 goroutine(线程)发送通知。
    • func (c *Cond) Broadcast()
      广播通知,给正在等待(阻塞)在该条件变量上的所有 goroutine(线程)发送通知。
  • 使用流程:
    1. 创建条件变量:var cond sync.Cond
    2. 指定条件变量用的锁:cond.L = new(sync.Mutex)
    3. cond.Lock() 给公共区加锁(互斥量)
    4. 判断是否达到阻塞条件(缓冲区满/空) --- for 循环判断
    5. 访问公共区 ---读、写数据、打印
    6. 解锁条件变量用的锁 cond.L.Unlock()
    7. 唤醒阻塞在条件变量上的对端。signal() Broadcast()
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351

推荐阅读更多精彩内容