golang 基础(21)协程

zidea · · 572 次点击 · · 开始浏览    
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

golang_real.jpg

并发编程

在函数调用时在函数前面加一个关键字 go 这个函数就会并发执行,

func main(){
    for i:= 0; i < 10; i++{
        go func(i int){
            for {
                fmt.Printf("hello from goroutine %d\n", i)
            }
        }(i)
    }
}

现在执行是没有效果,因为我们goroutine 是并发执行的,但是没有等到 gorutine 进行打印我们 main 就结束了,main 结束了所有的协程也就结束了。

for i:= 0; i < 10; i++{
        go func(i int){
            for {
                fmt.Printf("hello from goroutine %d\n", i)
            }
        }(i)
    }

time.Sleep(time.Millisecond)
hello from goroutine 535
hello from goroutine 535
hello from goroutine 535
hello from goroutine 535
hello from goroutine 535
hello from goroutine 535
hello from goroutine 535
hello from goroutine 535
golang-project-structure.jpg

协程

  • 轻量级线程
  • 非抢占式多任务处理,由协程主动交出控制权
    线程是没有控制权,有时候线程执行一半就有可能被系统切掉,然后再返回线程继续执行。而协程则不然,主动权掌握在自己手里,正式因为是非抢占式所以才能轻量级,因为线程被中断执行时,就需要记录很多东西以备切换回来后还能正常执行,所以线程就相对于协程比较重。
  • 编译器、解释器或者是虚拟机层面的多任务
  • 多个协程可能在一个或多个线程上运行

上面的代码我们看输出好像和我们做 java 的线程 demo 差不多,这是因为在每个 goroutine 中都进行io操作,每个 io 操作都是耗时的,我们换一下,让 goroutine 更新一个数组

func main(){
    var a [10]int
    for i:= 0; i < 10; i++{
        go func(i int){
            for {
                a[i]++
                // fmt.Printf("hello from goroutine %d\n", i)
            }
        }(i)
    }


    time.Sleep(time.Millisecond)
    fmt.Println(a)
}

然后执行完毕打印一下这个数组,之前的 io 操作会有协程的切换这里 a[i]++ 会这个对应内存块数字进行加1,这样不会有等待,所以就会被一个协程抢掉。
因为在这个协程中 for 是无限循环所以无法交出控制权,而 main 函数自己也是一个协程,所以是无法得到控制权来执行下面的代码的

解决方案是我们手动交出控制权

go func(i int){
        for {
            a[i]++
            // fmt.Printf("hello from goroutine %d\n", i)
            runtime.Gosched()
        }
    }(i)
golang_drink.jpg

有疑问加站长微信联系(非本文作者)

本文来自:简书

感谢作者:zidea

查看原文:golang 基础(21)协程

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

572 次点击  
加入收藏 微博
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传