golang 基础(21)协程

作者: zidea | 来源:发表于2019-03-29 19:05 被阅读11次
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

相关文章

  • golang 基础(21)协程

    并发编程 在函数调用时在函数前面加一个关键字 go 这个函数就会并发执行, 现在执行是没有效果,因为我们gorou...

  • 浅谈GoLang协程

    GoLang协程 学习golang也有一段时间了,这里讲一下自己对golang协程的使用理解,golang很多人都...

  • go 中 限制 goroutine 数量以及使用协程池

    需要使用协程池? Golang 开发需要协程池吗[https://www.zhihu.com/question/3...

  • golang中最大协程数的限制(线程)

    golang中最大协程数的限制 golang中有最大协程数的限制吗?如果有的话,是通过什么参数控制呢?还是通过每个...

  • Golang:协程

    协程(Goroutine)是与其他函数或方法同时运行的函数或方法。可以认为它是轻量级的线程。与线程相比,创建 Go...

  • 协程--golang

    协程(goroutine) 是轻量级的执行线程,由关键字go定义 一个例子 特别注意:如果没有time.Sleep...

  • AndroidのKotlin协程

    参考资料:协程基础 1.协程Coroutines基础 1.1 GlobalScope.launch启动一个独立协程...

  • 协程的使用

    Golang原生支持协程。只需要在前面加上go 就可以启动协程,比如说: package mainimport("...

  • Golang源码之Goroutine

    原文来自博客园作者: q303248153 Golang最大的特色可以说是协程(goroutine)了, 协程让本...

  • Go语言学习笔记七(并发编程)

    协程机制 Golang 线程和协程的区别 备注:需要区分进程、线程(内核级线程)、协程(用户级线程)三个概念。 进...

网友评论

    本文标题:golang 基础(21)协程

    本文链接:https://www.haomeiwen.com/subject/relwmqtx.html