从零学习 Go 语言(24):理解 Go 语言中的 goroutine

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

![](http://image.iswbm.com/20200607145423.png) 在线博客:http://golang.iswbm.com/ Github:https://github.com/iswbm/GolangCodingTime --- 说到Go语言,很多没接触过它的人,对它的第一印象,一定是它从语言层面天生支持并发,非常方便,让开发者能快速写出高性能且易于理解的程序。 在 Python (为Py为例,主要是我比较熟悉,其他主流编程语言也类似)中,并发编程的门槛并不低,你要学习多进程,多线程,还要掌握各种支持并发的库 asyncio,aiohttp 等,同时你还要清楚它们之间的区别及优缺点,懂得在不同的场景选择不同的并发模式。 而 Golang 作为一门现代化的编程语言,它不需要你直面这些复杂的问题。在 Golang 里,你不需要学习如何创建进程池/线程池,也不需要知道什么情况下使用多线程,什么时候使用多进程。因为你没得选,也不需要选,它原生提供的 goroutine (也即协程)已经足够优秀,能够自动帮你处理好所有的事情,而你要做的只是执行它,就这么简单。 一个 goroutine 本身就是一个函数,当你直接调用时,它就是一个普通函数,如果你在调用前加一个关键字 `go` ,那你就开启了一个 goroutine。 ```go // 执行一个函数 func() // 开启一个协程执行这个函数 go func() ``` ## 1. 协程的初步使用 一个 Go 程序的入口通常是 main 函数,程序启动后,main 函数最先运行,我们称之为 `main goroutine`。 在 main 中或者其下调用的代码中才可以使用 `go + func()` 的方法来启动协程。 main 的地位相当于主线程,当 main 函数执行完成后,这个线程也就终结了,其下的运行着的所有协程也不管代码是不是还在跑,也得乖乖退出。 因此如下这段代码运行完,只会输出 `hello, world` ,而不会输出`hello, go`(因为协程的创建需要时间,当 `hello, world`打印后,协程还没来得及并执行) ```go import "fmt" func mytest() { fmt.Println("hello, go") } func main() { // 启动一个协程 go mytest() fmt.Println("hello, world") } ``` 对于刚学习Go的协程同学来说,可以使用 time.Sleep 来使 main 阻塞,使其他协程能够有机会运行完全,但你要注意的是,这并不是推荐的方式(后续我们会学习其他更优雅的方式)。 当我在代码中加入一行 time.Sleep 输出就符合预期了。 ```go import ( "fmt" "time" ) func mytest() { fmt.Println("hello, go") } func main() { go mytest() fmt.Println("hello, world") time.Sleep(time.Second) } ``` 输出如下 ``` hello, world hello, go ``` ## 2. 多个协程的效果 为了让你看到并发的效果,这里举个最简单的例子 ```go import ( "fmt" "time" ) func mygo(name string) { for i := 0; i < 10; i++ { fmt.Printf("In goroutine %s\n", name) // 为了避免第一个协程执行过快,观察不到并发的效果,加个休眠 time.Sleep(10 * time.Millisecond) } } func main() { go mygo("协程1号") // 第一个协程 go mygo("协程2号") // 第二个协程 time.Sleep(time.Second) } ``` 输出如下,可以观察到两个协程就如两个线程一样,并发执行 ``` In goroutine 协程2号 In goroutine 协程1号 In goroutine 协程1号 In goroutine 协程2号 In goroutine 协程2号 In goroutine 协程1号 In goroutine 协程1号 In goroutine 协程2号 In goroutine 协程1号 In goroutine 协程2号 In goroutine 协程1号 In goroutine 协程2号 In goroutine 协程1号 In goroutine 协程2号 In goroutine 协程1号 In goroutine 协程2号 In goroutine 协程1号 In goroutine 协程2号 In goroutine 协程1号 In goroutine 协程2号 ``` 通过以上简单的例子,是不是折服于Go的这种强大的并发特性,将同步代码转为异步代码,真的只要一个关键字就可以了,也不需要使用其他库,简单方便。 本篇只介绍了协程的简单使用,真正的并发程序还是要结合 信道 (channel)来实现。关于信道的内容,将在下一篇文章中介绍。 ## 系列导读 --- [从零学习 Go 语言(01):一文搞定开发环境的搭建](https://studygolang.com/articles/27365) [从零学习 Go 语言(02):学习五种变量创建的方法](https://studygolang.com/articles/27432) [从零学习 Go 语言(03):数据类型之整型与浮点型](https://studygolang.com/articles/27440) [从零学习 Go 语言(04):byte、rune与字符串](https://studygolang.com/articles/27463) [从零学习 Go 语言(05):数据类型之数组与切片](https://studygolang.com/articles/27508) [从零学习 Go 语言(06):数据类型之字典与布尔类型](https://studygolang.com/articles/27563) [从零学习 Go 语言(07):数据类型之指针](https://studygolang.com/articles/27585) [从零学习 Go 语言(08):流程控制之if-else](https://studygolang.com/articles/27613) [从零学习 Go 语言(09):流程控制之switch-case](https://studygolang.com/articles/27660) [从零学习 Go 语言(10):流程控制之for 循环](https://studygolang.com/articles/28120) [从零学习 Go 语言(11):goto 无条件跳转](https://studygolang.com/articles/28472) [从零学习 Go 语言(12):流程控制之defer 延迟语句](https://studygolang.com/articles/28515) [从零学习 Go 语言(13):异常机制 panic 和 recover](https://studygolang.com/articles/28519) [从零学习 Go 语言(14):Go 语言中的类型断言是什么?](https://studygolang.com/articles/29305) [从零学习 Go 语言(15):学习 Go 语言的结构体与继承](https://studygolang.com/articles/29306) [从零学习 Go 语言(17):Go 语言中的 make 和 new 有什么区别?](https://studygolang.com/articles/29315) [从零学习 Go 语言(18):Go 语言中的 语句块与作用域](https://studygolang.com/articles/29365) [从零学习 Go 语言(19):Go Modules 前世今生及入门使用](https://studygolang.com/articles/29371) [从零学习 Go 语言(20):关于包导入必学的 8 个知识点](https://studygolang.com/articles/29404) [从零学习 Go 语言(21):一文了解 Go语言中编码规范](https://studygolang.com/articles/29477) [从零学习 Go 语言(22):Go 语言中如何开源自己写的包给别人用?](https://studygolang.com/articles/29609) [从零学习 Go 语言(23):一篇文章搞懂 Go 语言的函数](https://studygolang.com/articles/29628) --- ![](http://image.python-online.cn/20200321153457.png)

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

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

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