当前位置: 首页 > 面试题库 >

去调度程序和CGO:请解释这种行为差异?

桂浩言
2023-03-14
问题内容

我想知道实现的原因:

package main

func main() {
    c := make(chan struct{})

    go func() {
        print("a")
        for {
        }
    }()

    go func() {
        print("b")
        for {
        }
    }()

    go func() {
        print("c")
        c <- struct{}{}
        for {
        }
    }()

    <-c
}

❯❯❯ GOMAXPROCS=2 go run sample1.go                                    
ab  <--- blocks.
package main

// static void loop() { for(;;); }
import "C"

func main() {
    c := make(chan struct{})

    go func() {
        print("a")
        C.loop()
        print("x")
    }()

    go func() {
        print("b")
        C.loop()
        print("y")
    }()

    go func() {
        print("c")
        c <- struct{}{}
        C.loop()
        print("z")
    }()

    <-c
}

❯❯❯ GOMAXPROCS=2 go run sample2.go                                    
abc  <--- ends gracefully.

更具体地说,我的意思是C紧密循环与go例程调度中的Go循环有何不同。即使应该在Go程序结束时突然终止C紧密循环,我还是想知道依靠这种行为来启动C任务而不阻塞Go程序是否安全


问题答案:

运行时无法抢占真正的繁忙循环。没有调度点的CPU密集型循环必须位于其自己的线程中,其他goroutine才能运行。函数调用和通道发送或接收操作都产生。网络IO是异步调度的,文件IO具有其自己的线程。

通常设置GOMAXPROCS>
1就足够了,但是由于您有3个这样的循环,并且只有2个线程,因此调度程序仍然被阻塞。如果您有一个有效的CPU密集型循环,难以安排goroutine,则可以runtime.GoSched()定期调用以让出给调度程序。在现实世界中,唯一通常会发生这种情况的地方是编程错误,即您有一个空循环。

所有cgo调用都在go运行时之外的它们自己的线程中发生,因此这些循环除了浪费CPU外,对主循环没有任何影响。



 类似资料:
  • 问题内容: 我想使用以下c作为Go的cgo: 建立: 我为Go的cgo重新编写了该代码: 尝试编译为: 但是我收到链接器错误: /tmp/go- build076004816/opensource.stdk/lib/tools/_obj/x11.cgo2.o:在函数XScreenSaverAllocInfo中/tmp/go- build076004816/opensource.stdk/lib/t

  • 我正在用一个项目中相互协作的两个独立的Web应用程序来试验一些问题。在进行了X次部署之后,我得到了臭名昭著的“java.lang.OutofMemoryError:PermGen Space”错误。 所以我已经用VisualVM监视PermGen空间一段时间了,不断地重新部署应用程序,看看发生了什么。 这里有奇怪的行为: 首先,我已经重新部署了15倍以上的第一个应用程序。行为与预期的一样:内存图就

  • 请解释为什么输出 是。

  • 本文向大家介绍请你解释为什么会出现4.0-3.6=0.40000001这种现象?相关面试题,主要包含被问及请你解释为什么会出现4.0-3.6=0.40000001这种现象?时的应答技巧和注意事项,需要的朋友参考一下 考察点:计算机基础 原因简单来说是这样:2进制的小数无法精确的表达10进制小数,计算机在计算10进制小数的过程中要先转换为2进制进行计算,这个过程中出现了误差。

  • 我使用RxJava2 Android网络的网络调用。我面临的问题是,当我试图通过命中API时,有时它不会给出任何响应,而当我试图通过命中API时,它总是给出回应 和

  • 三个元素 前两个居中 第三个靠右显示