zwvista

导航

Go语言学习笔记(4)并发

Go 例程(Goroutines)

Go 例程是一种绿色线程,使用关键字 go 来启动。

https://gobyexample.com/goroutines

func say(s string) {
    for i := 0; i < 5; i++ {
        time.Sleep(100 * time.Millisecond)
        fmt.Println(s)
    }
}
go say("world")
say("hello")
/*
world
hello
hello
world
hello
world
hello
world
hello
*/

信道(Channels)

// 无缓冲的信道的创建,发送和接收
ch := make(chan int)
ch <- v    // Send v to channel ch.
v := <-ch  // Receive from ch, and
           // assign value to v.
// Go 例程 + 信道
func sum(s []int, c chan int) {
    sum := 0
    for _, v := range s {
        sum += v
    }
    c <- sum // send sum to c
}
s := []int{7, 2, 8, -9, 4, 0}
c := make(chan int)
go sum(s[:len(s)/2], c)
go sum(s[len(s)/2:], c)
x, y := <-c, <-c // receive from c
fmt.Println(x, y, x+y) // -5 17 12
// 有缓冲的信道的创建,发送和接收
ch := make(chan int, 2)
ch <- 1 // send
ch <- 2 // send
// ch <- 3 // send // error
fmt.Println(<-ch) // receive
fmt.Println(<-ch) // receive
// ch <- 3 // send // ok
// 单向信道
// Only for receiving
mychanl1 := make(<-chan string)
// Only for sending
mychanl2 := make(chan<- string)
// 单向信道通常用作Go例程形参类型
func pong(pings <-chan string, pongs chan<- string) {
    msg := <-pings
    pongs <- msg
}
  • 信道是(Go例程之间)用来通信的管道
  • 信道是类型安全的
  • 信道的数据结构是先进先出的队列
  • 无缓冲的信道是同步的,发送和接收必须同时进行,不然会导致堵塞
  • 有缓冲的信道是异步的,发送和接收不必同时进行
  • 单向信道可以进一步增强安全性

遍历信道,关闭信道

// 调用 close 函数关闭信道
// 使用 for 循环遍历信道
func fibonacci(n int, c chan int) {
    x, y := 0, 1
    for i := 0; i < n; i++ {
        c <- x // send
        x, y = y, x+y
    }
    close(c)
}
c := make(chan int, 10)
go fibonacci(cap(c), c)
for i := range c { // receive
    fmt.Println(i)
}
/*
0
1
1
2
3
5
8
13
21
34
*/

选择信道

使用 select 语句可以同时等待多个信道的发送和接收操作。

// 使用 select 语句实现线程同步
func fibonacci(c, quit chan int) {
    x, y := 0, 1
    for {
        select {
        case c <- x: // send
            x, y = y, x+y
        case <-quit: // receive
            fmt.Println("quit")
            return
        }
    }
}
c := make(chan int)
quit := make(chan int)
go func() {
    for i := 0; i < 10; i++ {
        fmt.Println(<-c) // receive
    }
    quit <- 0 // send
}()
fibonacci(c, quit)
/*
0
1
1
2
3
5
8
13
21
34
quit
*/
// 使用 select 语句实现 Timeout
tick := time.Tick(100 * time.Millisecond) // send
boom := time.After(500 * time.Millisecond) // send
for {
    select {
    case <-tick: // receive
        fmt.Println("tick.")
    case <-boom: // receive
        fmt.Println("BOOM!")
        return
    default:
        fmt.Println("    .")
        time.Sleep(50 * time.Millisecond)
    }
}
/*
    .
    .
tick.
    .
    .
tick.
    .
    .
tick.
    .
    .
tick.
    .
    .
BOOM!
*/

sync.Mutex

排它锁

// SafeCounter is safe to use concurrently.
type SafeCounter struct {
    v   map[string]int
    mux sync.Mutex
}
// Inc increments the counter for the given key.
func (c *SafeCounter) Inc(key string) {
    c.mux.Lock()
    // Lock so only one goroutine at a time can access the map c.v.
    c.v[key]++
    c.mux.Unlock()
}
// Value returns the current value of the counter for the given key.
func (c *SafeCounter) Value(key string) int {
    c.mux.Lock()
    // Lock so only one goroutine at a time can access the map c.v.
    defer c.mux.Unlock()
    return c.v[key]
}
c := SafeCounter{v: make(map[string]int)}
for i := 0; i < 1000; i++ {
    go c.Inc("somekey")
}
time.Sleep(time.Second)
fmt.Println(c.Value("somekey")) // 1000

sync.WaitGroup

线程间同步

func process(i int, wg *sync.WaitGroup) {  
    fmt.Println("started Goroutine ", i)
    time.Sleep(2 * time.Second)
    fmt.Printf("Goroutine %d ended\n", i)
    wg.Done()
}
var wg sync.WaitGroup
for i := 0; i < 3; i++ {
    wg.Add(1)
    go process(i, &wg)
}
wg.Wait()
fmt.Println("All go routines finished executing")
/*
started Goroutine  2
started Goroutine  0
started Goroutine  1
Goroutine 2 ended
Goroutine 1 ended
Goroutine 0 ended
All go routines finished executing
*/

参考链接

Is a Go goroutine a coroutine?

posted on 2020-05-14 13:40  zwvista  阅读(249)  评论(0编辑  收藏  举报