go语言并发小问题

afyji · 2014-12-18 15:06:00 · 3948 次点击 · 大约8小时之前 开始浏览    置顶
这是一个创建于 2014-12-18 15:06:00 的主题,其中的信息可能已经有所发展或是发生改变。

package main

import (
    "fmt"
)

func loop(silice []int, c chan int, res string) {
    sum := 0
    for _, v := range silice {
        sum += v
    }
    c <- sum
    fmt.Println(res, "---sum:", sum)
}

func main() {
    c := make(chan int)
    arr := []int{1, 2, 3, 4, 5, 6}
    go loop(arr[:len(arr)/2], c, "first loop")
    go loop(arr[len(arr)/2:len(arr)], c, "second loop")

    x, y := <-c, <-c

    fmt.Println("x,y,x+y:", x, y, x+y)
}

输出结果为

first loop ---sum: 6
x,y,x+y: 6 15 21

起的函数名称不恰当。。。改的代码

为什么只会输出first,而不输出second,当执行两个go的时候,按说两个loop函数都会被挂起(阻塞),在主线执行时,第一个loop和第二个分别解除阻塞,那么为什么只有第一个loop 最后的print被输出,而第二个没有。

我的理解是,如果主线获取channel的取值后,他的过程是什么,是完成之前被阻塞的线程goroutine(那么 first和second都会被执行)?还是不再跳回之前的线程,继续然后执行主线(那么first和second 的两个输出应该都不会被执行)

可是现在输出了一个,只有first,让我有些费解,请个位同学赐教


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

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

3948 次点击  
加入收藏 微博
5 回复  |  直到 2014-12-21 16:34:09
qq716683293
qq716683293 · #1 · 10年之前

算上main是有三个go协程。第一次获得数据会正常输出。然后main取出管道中值, 第二次main接到管道中的数据后main直接结束。第二个函数里的输出语句即将打印但还没有打印。 这时候main退出会导致第二个协程死掉。所以就看不到输出了。

qkb_75_go
qkb_75_go · #2 · 10年之前

这段代码,如去掉 x, y := <-c, <-c,则任何 goroutine 都不会执行就退出了。 这段代码,如加上 runtime.GOMAXPROC(2)调度到两个 core 上,又不一样。 不觉得有什么异常呀?

fsyx
fsyx · #3 · 10年之前

c <- sum fmt.Println(res, "---sum:", sum) 将这两段代码的位置调换一下就OK了

afyji
afyji · #4 · 10年之前

这就是我的问题,main和loop是在同时执行的么,因为如果是同时执行,那么我可以理解。对不起,我对并性刚开始接触。通过我对并性和并发的理解。这里面在执行loop和在执行main的时候,同一时间只有一个在执行,那么当从main中跳转到第二个loop的时候,main应该是挂起的,也就是或不应该会出现main执行结束而第二个loop没有执行完啊。

cloudaice
cloudaice · #5 · 10年之前
afyjiafyji #4 回复

这就是我的问题,main和loop是在同时执行的么,因为如果是同时执行,那么我可以理解。对不起,我对并性刚开始接触。通过我对并性和并发的理解。这里面在执行loop和在执行main的时候,同一时间只有一个在执行,那么当从main中跳转到第二个loop的时候,main应该是挂起的,也就是或不应该会出现main执行结束而第二个loop没有执行完啊。

这个问题很好理解,如果你在最后sleep一下,肯定可以输出的,因此直接的原因是主程序退出了。导致没有打印出来。本质上原因是你没有做好同步,对于不带buffer的channel来说,读在写之前发生。这是Go语言同步的一条原则。

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