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

2020-04-20  本文已影响0人  梁坤同学

select

超时处理

有时候会出现 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
}

锁和条件变量

什么是锁

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

死锁

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

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

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 既不能进行读操作,也不能进行写操作。

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

条件变量

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

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

type Cond struct {
  noCopy noCopy
  L Locker
  notify notifyList
  checker copyChecker
}
上一篇 下一篇

猜你喜欢

热点阅读