- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
拿这段代码:
func main() {
var x int
go func() {
for {
x++
}
}()
time.Sleep(time.Second)
fmt.Println("x =", x)
}
为什么 x
最后等于 0
?我知道 Go 的调度程序需要 time.Sleep()
调用来获取 goroutine,但为什么它不这样做?
提示:将 time.Sleep()
或对 runtime.Gosched()
的调用放在里面for 循环修复了这段代码。但是为什么?
更新:检查相同代码的以下版本:
func main() {
var x int
go func() {
for i := 0; i < 10000; i++ {
x++
}
}()
time.Sleep(time.Second)
fmt.Println("x =", x)
}
奇怪的是goroutine里面的代码现在执行完了,x
已经不是0了,编译器有没有优化这里?
最佳答案
了解您在此处询问的内容很重要。 Go 中没有 promise 这个程序会做任何特别的事情,因为这个程序是无效的。但作为对优化器的探索,深入了解它目前是如何实现的可能会很有趣。任何依赖此信息的程序都将非常脆弱和无效,但这仍然是一个好奇心。
我们可以编译程序,然后查看输出。我特别喜欢你提供的两个版本,因为它们让用户看到了差异。我已经使用 Hopper 完成了反编译(这些是使用 go1.14 darwin/amd64 编译的)。
在第二种情况下,goroutine 看起来像你认为的那样:
void _main.main.func1(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6) {
rax = arg6;
for (rcx = 0x0; rcx < 0x2710; rcx = rcx + 0x1) {
*rax = *rax + 0x1;
}
return;
}
这里没有什么太令人惊讶的。但是您好奇的第一个案例呢:
_main.main.func1:
goto _main.main.func1;
它变成了一个空洞。毫不夸张的说;这是程序集:
_main.main.func1:
000000000109d1b0 nop ; CODE XREF=_main.main.func1+1
000000000109d1b1 jmp _main.main.func1 ; _main.main.func1
这是怎么发生的?好了,编译器可以看看这段代码:
go func() {
for {
x++
}
}()
它知道没有任何东西会读取x
。任何东西都无法读取 x
,因为 x
周围没有锁定,并且这个 goroutine 永远不会终止。因此,在这个 goroutine 完成后,没有任何东西可以读取 x
。参见 The Go Memory Model详细了解某事发生在另一件事之前或之后的含义。
“但我确实读过 x!”不,你不知道。那将是无效代码,编译器知道您没有编写无效代码。当有竞争检测器告诉您这是无效的时,谁会这样做?因此,由于编译器可以清楚地看到没有任何东西读取过 x
,因此没有理由去更新它。
在您的有限循环示例中,goroutine 终止,因此之后可能会读取 x
。编译器不够聪明,无法注意到从未进行过有效 读取,因此它没有尽可能优化它。也许 future 的编译器会足够聪明,在两种情况下都输出 0。也许 future 的编译器会足够聪明,可以在第一种情况下完全删除您的无操作 goroutine。
但这里的关键点是无限循环的情况是完全正确的,尽管效率比它可能的要低一些。非无限循环的情况也是完全正确的,尽管效率要低得多。
关于go - 为什么这个 goroutine 不运行,即使是 `time.Sleep` ?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61017203/
我在 linux (Centos) 上使用 pthread 编程?我想让线程休眠一小段时间以等待某些事情。我正在尝试使用 sleep()、nanosleep() 或 usleep() 或其他可以做到这
此要求出现在我的 Android 应用程序中,但它通常适用于 Java。我的应用程序每隔几秒钟“做某事”。我已经按如下方式实现了这一点(只是相关的 fragment - 不是完整的代码): fragm
我正在使用 esp8266 构建 IR 到 WiFi 桥接器。基本上,我正在构建一个连接到红外 Remote 内的 esp8266 的红外接收器,以通过 wifi 将接收到的红外远程按键转发到服务器。
我想让 Toast 出现,然后让 sleep 运行。 如果我这样做,Toast 会在 sleep 后出现,但我希望反过来。有人有建议吗?这是我的代码 switch (checkedRadioButto
我在做一件简单的事情,打印一条错误消息,延迟执行 5 秒,然后调用另一个函数,这是代码 public void saveAndDisplay() throws InterruptedException
我的 Activity 上有一个按钮,当我单击它时,我希望按钮改变颜色,等待一段时间,然后再次改变颜色。 我尝试过以下两个版本的 sleep : 尝试1: public void buClick(Vi
我尝试过这个: for(int i =0; i1。创建ScheduledExecutorService public static ScheduledExecutorService createSch
我有一个 Winform,需要等待大约 3 - 4 小时。我无法关闭并以某种方式重新打开应用程序,因为它在等待时在后台做的事情很少。 为了实现等待 - 不给 UI 线程造成麻烦和其他原因 - 我有一个
我在网上看到了下面一段关于 Linux 线程的代码。但是当我运行它时,所有线程似乎都在 sleep ,而不仅仅是主线程。为什么?另外,如果没有 sleep(5),“线程创建成功”语句会运行 3 次而不
我有一个 php 脚本,我需要每 5 秒运行一次(运行,等待它完成,等待 5 秒,再次运行) 我有两种方法。要么在脚本中有一个带有 sleep 功能的无限循环,看起来像这样: while (1) {
我有一个图形用户界面,我想显示一些文本,然后稍等一下。 我的代码看起来像这样: //do something (add JTextArea, revalidate, repaint) try{
我想知道安卓手机的 sleep 模式和深度 sleep 模式有什么区别,手机进入休眠模式和深度 sleep 模式会有什么影响。 请提供详细的答案。 问候,皮克斯 最佳答案 深度 sleep 模式与休眠
我正在学习 C。在这个节目中我使用 sleep 功能来减慢倒计时。我的教科书没有指定我应该包含的库来使用 sleep 功能。所以我使用它时没有为它包含任何特殊的库并且它可以工作。但它在代码块中给了我这
我正在尝试模拟按键按下和按键 Action 。 例如:2638 毫秒。 SendMessage(hWnd, WM_KEYDOWN, keyCode, 0); Sleep(2638); SendMess
我在 while 循环中调用一个线程 hibernate 1 秒。当标志为真时,循环将运行(标志为真无限时间)。在循环线程内应该 hibernate 1 秒,唤醒并增加计数器,检查 IF 条件,在 F
有很多文件说“你应该避免使用带警报的 sleep ,因为许多系统使用警报来实现 sleep ”。实际上,我正在为这个问题而苦恼。 那么,当 sleep() 不能很好地处理警报时,是否有人可以帮助我“
我有两个带有图像的jlabel..我尝试在单击另一个标签时更改标签中的图像..(例如游戏)..我有一个问题..当我编写 Thread.sleep 时,图像没有改变..请参阅代码: public cla
我正在研究多线程,我有一个关于线程 sleep 方法的问题。当我在已经处于 sleep 线程(时间 t2)中执行 sleep()(时间 t1)方法时。总 sleep 时间为 t1+t2 或 t2(如果
如果我们不向 sleep( ) 函数传递任何参数,默认 sleep 时间是多少? #include int main() { int pid,dip,cpid; pid = fork(
当我转到 asyncio 页面时,第一个示例是一个 hello world 程序。当我在 python 3.73 上运行它时,我看不出与正常的有什么不同。 谁能告诉我区别并举一个重要的例子? In [
我是一名优秀的程序员,十分优秀!