- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我刚刚开始学习 Go 并遵循包含以下 goroutines 示例的教程:
package main
import (
"fmt"
"runtime"
)
func say(s string) {
for i := 0; i < 5; i++ {
runtime.Gosched()
fmt.Println(s)
}
}
func main() {
go say("world") // create a new goroutine
say("hello") // current goroutine
}
它声明“runtime.Gosched() 意味着让 CPU 执行其他 goroutine,并在某个时刻返回。”。示例下方给出了以下输出:
hello
world
hello
world
hello
world
hello
world
hello
然而,当我在我的机器上使用 go run
运行这个例子时,我得到了
hello
world
world
world
world
world
hello
hello
hello
hello
我的 Go 版本是 go version go1.6 darwin/amd64
。
其实这两个结果我都不明白!为什么不只是
hello
?据我了解 Go 程序在程序的最后一条语句执行后退出,所以我认为 say()
作为 goroutine 运行并且其执行被延迟后,程序执行下一个 say()
作为普通函数,打印“hello”然后退出。
那么哪个结果是正确的,为什么?
最佳答案
第一个输出将由单核机器生成。第二个可以由多核生成。
say
是一个内部有 for 循环的函数,迭代 5 次。它确实是一个普通函数,但其中调用了 Gosched
。 Gosched
所做的是,它告诉运行时暂停执行当前的 goroutine,而是启动另一个等待的 goroutine。这称为屈服。
解释第一个输出
这是您可以期望在单核机器上获得的输出。一步一步来,
go say("world")
在这一步,运行时开始在单独的 goroutine 上执行 say("world")
调用并继续主 goroutine。但是机器只有一个核心。所以两个 goroutines 不能并行运行。新的 goroutine(例如 gr A
)必须等到正在运行的主 goroutine(例如 gr B
)完成或暂停(yields)。所以它等待。主 goroutine 开始执行 say("hello")
现在通过 gr B
的 say
函数运行时遇到 runtime.Gosched()
Gosched
调用就像暂停。它告诉运行时暂停我并安排另一个正在等待的 goroutine。所以运行时调度 gr A
。它从它等待的地方开始,也就是,
说(“世界”)
现在 gr A
执行直到遇到它自己的 runtime.Gosched()
。 gr A
暂停。 gr B
醒来并从它离开的地方开始奔跑。 runtime.Gosched()
之后的语句是打印“hello”。所以“你好”被打印出来了。 gr B
继续并进入其 for 循环的下一次迭代。遇见 Gosched
。暂停。 gr A
重新启动。打印“世界”。我想您可以看到这是如何进行 5 次,直到它打印给定的输出。
解释第二个输出
如果你的机器有多个核心,goroutines 可以并行运行。你的是你得到的输出。
现在当 go say("world")
被调用时 gr A
不必等到 gr B
完成。它可以立即在另一个核心上启动。因此,当 Gosched
被调用时,可能没有等待的 goroutines。如果当前的暂停,它将立即在另一个核心上启动。
所以在多核机器中你不能保证单词的打印顺序。如果您多次运行该程序,我想您也会看到其他命令。
您可以将 GOMAXPROCs 设置为 1 并查看程序在单核机器上的运行情况。
func main() {
runtime.GOMAXPROCS(1)
go say("world") // create a new goroutine
say("hello") // current goroutine
}
然后你会看到第一个输出。
关于需要 Goroutine 示例解释,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36261502/
发件人:http://blog.nindalf.com/how-goroutines-work/ As the goroutines are scheduled cooperatively, a go
很多时候在用 Go 开发 http 服务器时,我都会遇到这种困境。 假设我想尽快用http statuscode 200响应客户端(然后在后面执行工作),这就是我通常这样做的原因: 我让我的主要 ht
这是代码: import "fmt" func main() { messages := make(chan string, 1) go func(c chan string) {
我正在学习 Golang,但遇到了一些困难。我已经研究过 Google,但没有任何进展。 我编写了一个代码,通过多台服务器的 ICMP 检查 RTT。 它有这样的结构: type Server str
我想运行多个 goroutine,进行一些处理,将结果放入 channel ,当至少有一个 goroutine 完成时,完成所有其他 goroutine 并从 channel 返回结果。 所以,我尝试
我有两个(但以后我会是三个)go 例程来处理来自远程服务器(来自 ampq channel )的传入消息。但是因为它们正在处理相同的数据/状态,所以我想阻止所有其他 go 例程,除了正在运行的例程。
我有一个案例,我从 2 个不同的位置(ES 和 REDIS)读取数据,我需要从最快的源读取一个值,因此我触发了 2 个 goroutines,一个从 ES 获取数据,其他从REDIS获取。 一旦从其中
像这里一样,我创建了一个 go playground 示例:sGgxEh40ev ,但无法正常工作。 quit := make(chan bool) res := make(chan int) go
我是golang的新手,正在研究goroutine。 我写了一个简单的代码,故意使用 goroutine 来划分数字。 首先,我给出基数并继续除它的数,直到它不能被整除 但是,我改变了go split
Main { go routine_1(carryout a time consuming task and return output) go routine_2(wait for output f
我想知道从另一个 goroutine 返回时调用的 goroutine 会发生什么。他们是继续运行还是被终止?这是一个示例代码来说明我的意思: func func() { // Doing s
更具体地说,在我的例子中,我有一个网络服务器和一个全局可访问的结构,网络服务器使用它来生成页面。我有另一个 Goroutine,它总是定期用新值更新该结构。这会引起问题吗?我是否需要实现一种机制来确保
来自 this file ,我不明白为什么函数startWorker会这样写: func (p *WorkerPool) dispatch() { for i := 0; i < p.maxW
我正在学习围棋,但在使用 goroutines 时遇到了问题。这是我的代码 package main import ( "fmt" "sync" "time" ) var co
我收到以下错误,我不明白为什么: 发送:查询 Herefatal 错误:所有 goroutines 都睡着了 - 死锁! 您可以看到我正在调用我使用 goroutine 创建的函数 routine。我
大家好,我正在从 Python3 过渡到 Go,所以我正在尝试重写我创建的库以获得更好的性能。 我面临一个问题,因为我是 Golang XD 中的新手,我使用有限的 API 下载数百个 json,我想
我有以下格式的脚本部分: func main() { for i=0;i<1000000;i++ { go test() } } func test() { a := test
package main func main() { c:=make(chan int) for i:=0; i<=100;i++ {
我正在学习 Go,我的第一个项目是一个简单的 ping 脚本。本质上,我想 ping 一堆 url,并在每个响应时等待 XXX 秒,然后再次 ping。这是删减的代码: func mai
这个问题在这里已经有了答案: Go all goroutines are asleep deadlock (2 个回答) fatal error: all goroutines are asleep
我是一名优秀的程序员,十分优秀!