• 很多人都是冲着 Go 大肆宣扬的高并发而忍不住跃跃欲试,但其实从源码的解析来看,goroutine 只是由官方实现的超级“线程池”而已。不过话说回来,每个实例 4-5KB 的栈内存占用和由于实现机制而大幅减少的创建和销毁开销,是制造 Go 号称的高并发的根本原因。另外,goroutine 的简单易用,也在语言层面上给予了开发者巨大的便利。
  • 并发不是并行:Concurrency Is Not Parallelism
  • 并发主要由切换时间片来实现“同时”运行,在并行则是直接利用多核实现多线程的运行,但 Go 可以设置使用核数,以发挥多核计算机的能力。
  • Goroutine 奉行通过通信来共享内存,而不是共享内存来通信。

​Channel​

  • Channel 是 goroutine 沟通的桥梁,大都是阻塞同步的
  • 通过 make 创建,close 关闭
  • Channel 是引用类型
  • 可以使用 for range 来迭代不断操作 channel
  • 可以设置单向或双向通道
  • 可以设置缓存大小,在未被填满前不会发生阻塞

通过channel计算10次,累加

方法一:

package main

import (
"fmt"
"runtime"
)

func main() {
fmt.Println("NumCPU:", runtime.NumCPU())
//获取cpu核心数量
runtime.GOMAXPROCS(runtime.NumCPU())
// channel bool类型 缓存长度为10
c := make(chan bool, 10)
for i := 0; i < 10; i++ {
go Go(c, i)
}
for i := 0; i < 10; i++ {
<-c
}
}

func Go(c chan bool, index int) {
a := 1
for i := 0; i < 100000000; i++ {
a += i
}
fmt.Println(index, a)
c <- true

输出结果:

Go 并发concurrency 学习笔记_make

方法二:

package main

import (
"fmt"
"runtime"
"sync"
)

func main() {
runtime.GOMAXPROCS(runtime.NumCPU())
wg := sync.WaitGroup{}
wg.Add(10)
for i := 0; i < 10; i++ {
go Go(&wg, i)
}
wg.Wait()
}

func Go(wg *sync.WaitGroup, index int) {
a := 1
for i := 0; i < 100000000; i++ {
a += i
}
fmt.Println(index, a)
wg.Done()
}

输出结果:

Go 并发concurrency 学习笔记_make_02

​Select​

  • 可处理一个或多个 channel 的发送与接收
  • 同时有多个可用的 channel时按随机顺序处理
  • 可用空的 select 来阻塞 main 函数
  • 可设置超时

1、处理多个 channel

package main

import (
"fmt"
)

func main() {
c1, c2 := make(chan int), make(chan string)
o := make(chan bool, 2)
go func() {
for {
select {
case v, ok := <-c1:
if !ok {
o <- true
break
}
fmt.Println("c1", v)
case v, ok := <-c2:
if !ok {
o <- true
break
}
fmt.Println("c2", v)
}
}

}()

c1 <- 1
c2 <- "hhhh"

c1 <- 9
c2 <- "uuuuu"

close(c1)
close(c2)

for i := 0; i < 2; i++ {
<-o
}
}

输出结果:

Go 并发concurrency 学习笔记_并发_03

2、同时有多个可用的 channel时按随机顺序处理

package main

import (
"fmt"
)

func main() {
o := make(chan int)
go func() {
for v := range o {
fmt.Println(v)
}

}()

for {
select {
case o <- 1:
case o <- 2:
}
}
}

输出结果:

Go 并发concurrency 学习笔记_go_04

3、超时

package main

import (
"fmt"
"time"
)

func main() {
o := make(chan int)

select {
case v := <-o:
fmt.Println(v)
case <-time.After(3 * time.Second):
fmt.Println("TimeOut")
}

}

输出结果:

Go 并发concurrency 学习笔记_make_05


个人微信公众号:

Go 并发concurrency 学习笔记_select_06

作者:jiankunking