golang并发总结

golang并发模型

  • go在语言层面提供了内置的并发支持
  • 不要通过共享内存来通信,而应该通过通信来共享内存


并发与并行

  • 定义
    • 并发: 指同一时刻, 系统通过调度,来回切换交替的运行多个任务,看起来是"同时"进行的.一个处理器同时处理多个任务
    • 并行:指同一时刻,2个任务"真正的"同时进行.多个处理器或者多核的处理器同时处理多个不同的任务.
  • 并行:多核cpu,物理上的同时执行.
image
  • 并发:同一时刻,只能一条指令执行.通过快速轮换执行,宏观上是多个线程同时执行的.微观上不是同时执行的,只是把时间分成若干段,使得多个线程快速交替执行.(单核cpu,逻辑上的同时执行)
image

常见的并发编程模型

  • 进程&线程(Apache)
最初的web服务器都是基于进程和线程,比如Apache,新到的一个请求就会分配一个进程或者线程,每个进程只服务一个用户,早期的互联网还不够普及,用户也不够多,这时候网站是可以稳定的,问题是
进程很昂贵,一台服务器无法创建很多的进程,后来随着互联网的发展,用户越来越多,
网站也变得越来越复杂,一个页面有可能就有上百个请求,
所以就诞生了C10K的问题,C10K的意思就是服务器同时支持一个10k量级的并发连接,就是要创建1w个进程,
这样的话,操作系统肯定是无法承受的.
所以进程和线程模型就显得很力不从心了.
  • 异步非阻塞(Nginx)
为了解决c10k的问题,就发明了异步非阻塞这种技术,一个典型的案例就是linux里的epoll,
一个普通的服务器就能服务大量的用户,资源消耗也很低,像NGINX都是epoll的产物,
但是,异步非阻塞也不是很完美,为了追求性能,强行的将线性程序打乱,开发和维护都变得非常复杂,
调试起来也比较困难.
  • 协程(Golang)
为了降低开发的复杂度,让程序员同学更爽的写代码,协程这种并发模型就逐渐流行起来,
这种模型,可以让我们像写线性程序一样,来写异步的程序,
其实协程的底层就是线程,但它比线程更轻量,几十个协程,体现在底层,可能也就是五六个的线程.
大家把协程理解成,更高效,更易用,更轻量的线程.

Glang并发的实现

  • 程序并发执行(goroutine)
每开启一个协程,就会有一个goroutine,负责程序的并发执行
f1() // 执行函数f1,等待函数f1返回

go f1() // 执行函数f1
f2()    // 不用等待f1()的返回

使用起来很简单, 只要在一个函数前面加 go 关键字就会创建一个goroutine, 去并发的执行,
所以程序并不会阻塞, 最后这2个函数相当于会去并发的执行.
有了goroutine之后就可以并发的去执行了,这就引出了一个问题:在多个goroutine之间是如何进行数据通信的呢?
比如 go f1() 和 f2() 这2个函数之间要通信,要传递数据的话,那他们是怎么进行的?

使用channel在多个goroutine之间进行数据通信和同步的
  • 多个goroutine间的数据同步和通信(channel)
    • channel的基础语法
      • 创建1:make(chan [type]) // 无缓冲
      • 创建2:make(chan [type], int) // 有缓冲
      • 写入:channel <-
      • 获取: <- channel
    c := make(chan string) // 声明一个无缓冲的channel
    // 创建一个goroutine
    go func() {  
        c <- "this is channel msg " // 发送数据到channel里
    }()
    msg := <-c // 阻塞直到接收到数据
    fmt.Println(msg)
    
说明:
// channel分为无缓冲信道(即unbuffered channel)和有缓冲信道(buffered channel)。
无缓冲的与有缓冲channel有着重大差别:一个是同步的 一个是非同步的
ch1:=make(chan int)        无缓冲
ch2:=make(chan int,1)      有缓冲
ch1<-1 // 无缓冲的
这里要有别的协程一直<-ch1 接受这个参数,
那么ch1<-1之后的代码才能执行,要不然就一直阻塞着,
ch2<-1 // 有缓冲的
这里则不会阻塞,因为缓冲大小是1(放了一个缓冲就剩0了),只有当放第二个的时候,第一个还没被拿走,这时候才会阻塞.
比喻:
    1. 无缓冲的就是一个送信人去你家门口送信,你不在家他不走,一定要送到你手里他才走.
    无缓冲保证信能到你手上.
    2. 有缓冲的就是一个送信人去你家门口送信,扔到你家信箱转身就走,除非你的信箱满了,他必须等信箱空下来.
    有缓冲保证信能到你家信箱
  • 多个channel选择数据读取或者写入(select)
使用select关键字,完成“多路选择”与“超时控制”。

    select {
    case v := <-ch1:
        fmt.Println("channel 1 msg =>", v)
    case v := <-ch2:
        fmt.Println("channel 2 msg =>", v)
    case <-time.After(time.Millisecond * 100): // 超时等待
        fmt.Println("time out")
    //default:
    //  fmt.Println("nothing")
    }
使用场景:可以监听写信号,进程的热启动,配置的热加载等 

协程的使用

func TestGroutine(t *testing.T) {
    for i := 0; i < 10; i++ {
        go func(i int) {
            fmt.Println(i) // 正确案例,值传递。各个协程无竞争关系。
        }(i)

        // go func() {
        //  fmt.Println(i) // 错误案例,共享变量。各个协程有竞争关系, 不安全
        // }()
    }
    time.Sleep(time.Millisecond * 50)
}

  • 协程并发,导致协程不安全
    // 协程不安全demo
    func TestThreadUnsafe(t *testing.T) {
        counter := 0
        for i := 0; i < 5000; i++ {
            go func() {
                counter++
            }()
        }
        time.Sleep(1 * time.Second)
        t.Logf("counter = %d", counter)
    }

    // 输出结果如下:
    === RUN   TestThreadUnsafe
    channel_test.go:346: counter = 4742 // 计算错误,因为并发导致了漏值
    --- PASS: TestThreadUnsafe (1.00s)
  • 如何保证协程安全?
    • 方式1: 普通加锁,并延迟等待协程执行完毕(不推荐)
    // 协程等待demo(停1秒,不推荐)
    func TestThreadSafe(t *testing.T) {
        var mut sync.Mutex // 互斥锁
        counter := 0
        for i := 0; i < 5000; i++ {
            go func() { // 开启协程
                defer func() {
                    mut.Unlock() //函数调用完成后:解锁,保证协程安全
                }()
                mut.Lock() // 函数将要调用前:加锁,保证协程安全
                counter++
            }()
        }
        time.Sleep(1 * time.Second) // 等待一秒,等协程全部执行完(如果程序复杂,1s可能不够用)
        t.Logf("counter = %d", counter)
    }
    // 输出结果如下:
    === RUN   TestThreadSafe
    channel_test.go:363: counter = 5000 
    // 结果正确,但是有一个问题。因为这里有个1秒的延迟等待,保证协程运行完毕再调用结果
    --- PASS: TestThreadSafe (1.00s)
  • 先来介绍下:同步等待组(WaitGroup)
waitGroup用于同步协程同步,等待一组协程执行完毕,才会继续向下执行.
1. 主协程调用Add()设置等待的协程数量.
2. 协程执行完毕,调用Done()函数
3. wait()函数阻塞,直到所有协程执行完毕才会继续向下执行.
  • 方法2 : 使用同步等待队列(waitGroup)保证顺序执行
    // 协程安全Demo
    func TestWaitGroup(t *testing.T) {
        var mut sync.Mutex    // 互斥锁
        var wg sync.WaitGroup // 等待队列
        counter := 0
        for i := 0; i < 5000; i++ {
            wg.Add(1) // 加个任务
            go func() {
                defer func() {
                    mut.Unlock() //函数调用完成后:解锁,保证协程安全
                }()
                mut.Lock() // 函数将要调用前:加锁,保证协程安全
                counter++
                wg.Done() // 做完任务
            }()
        }
        wg.Wait() //等待所有任务执行完毕
        t.Logf("counter = %d", counter)
    }
    // 运行结果如下:
    === RUN   TestWaitGroup
        channel_test.go:382: counter = 5000
    --- PASS: TestWaitGroup (0.00s)

channel的关闭和广播

  • close 内置函数关闭一个channel,该通道必须是双向的或仅发送的
ch1 := make(chan int, 1)
ch2 := make(chan<- int, 1)
ch3 := make(<-chan int, 1)
close(ch1)
close(ch2)
close(ch3) // 报错 invalid operation: close(ch3) (cannot close receive-only channel)
  • 向已经关闭的channel发送数据会panic
  • 关闭一个已经关闭的channel会panic
  • v, ok <- channel。 其中,ok为bool值,若ok == true时,表示channel处于open状态。 若ok==false时,表示channel处于close状态。

常见的并发场景

  • 只执行一次(单例模式)
    func TestOnceDo(t *testing.T) {
        var once sync.Once
        var wg sync.WaitGroup
        for i := 0; i < 10; i++ {
            wg.Add(1)
            // 在多协程的情况下,保证某段代码只执行一次。
            go func(ii int) {
                once.Do(func() {
                    t.Log(ii)
                })
                wg.Done()
            }(i)
        }
        wg.Wait()
    }
    // 输出结果
    === RUN   TestOnceDo
        channel_test.go:404: 0
    --- PASS: TestOnceDo (0.00s)
  • 发邮件,发短信
  • 跑脚本
  • 爬数据

总结

  • 关键字
    • go
    • make
    • chan
    • select
    • sync下的包
      • 互斥锁
      • 读写锁
      • waitGroup
      • map等
  • 将复杂的任务拆分, 让goroutine去并发的执行,通过channel做数据通信
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,843评论 6 502
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,538评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,187评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,264评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,289评论 6 390
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,231评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,116评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,945评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,367评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,581评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,754评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,458评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,068评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,692评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,842评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,797评论 2 369
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,654评论 2 354

推荐阅读更多精彩内容

  • 转载自:超详细的讲解Go中如何实现一个协程池 并发(并行),一直以来都是一个编程语言里的核心主题之一,也是被开发者...
    紫云02阅读 1,036评论 0 1
  • golang go和php的区别类型:go为编译性语言;php解释性语言错误:go的错误处理机制;php本身或者框...
    Impossible安徒生阅读 404评论 0 0
  • go并发编程入门到放弃 并发和并行 并发:一个处理器同时处理多个任务。 并行:多个处理器或者是多核的处理器同时处理...
    yangyunfeng阅读 562评论 0 2
  • 我是黑夜里大雨纷飞的人啊 1 “又到一年六月,有人笑有人哭,有人欢乐有人忧愁,有人惊喜有人失落,有的觉得收获满满有...
    陌忘宇阅读 8,535评论 28 53
  • 人工智能是什么?什么是人工智能?人工智能是未来发展的必然趋势吗?以后人工智能技术真的能达到电影里机器人的智能水平吗...
    ZLLZ阅读 3,777评论 0 5