为了账号安全,请及时绑定邮箱和手机立即绑定

从 goroutine 返回值会发生什么

从 goroutine 返回值会发生什么

Go
catspeake 2021-09-10 17:47:37
有人可以提供从 goroutine 返回的澄清值。goroutine 的返回值是否存储在权益上。例子 :// function getNumber returns the "int i" and we can't use this returned value// because this function is invoked as goroutine.// We know that, to communicate between main and goroutine one could// use the channel ( chan <- i), but I am interested to know about// the use of return i in goroutine. Is it possible to get/ use this // returned value.func getNumber(i int) int {    return i       }func main() {    for i:=0; i<10; i++ {        go printNumber(i)        }    time.Sleep(5)    }我们应该尽量避免 goroutine 中的返回值吗?
查看完整描述

3 回答

?
慕森王

TA贡献1777条经验 获得超3个赞

快速查看汇编输出显示


$ go build -gcflags -S z.go

该getNumber()函数确实将其结果存储到堆栈中


"".getNumber t=1 size=16 value=0 args=0x10 locals=0x0

    0x0000 00000 (z.go:5)   TEXT    "".getNumber+0(SB),4,$0-16

    0x0000 00000 (z.go:6)   MOVQ    "".i+8(FP),BX

    0x0005 00005 (z.go:6)   MOVQ    BX,"".~r1+16(FP)

    0x000a 00010 (z.go:6)   RET ,

所以当它被 goroutine 调用时,它确实将其结果存储到堆栈中。然而,这是一个新的堆栈,它在 goroutine 结束时被销毁,因此无法检索返回值。


"".main t=1 size=96 value=0 args=0x0 locals=0x18

    0x0000 00000 (z.go:9)   TEXT    "".main+0(SB),$24-0

    0x0000 00000 (z.go:9)   MOVQ    (TLS),CX

    0x0009 00009 (z.go:9)   CMPQ    SP,16(CX)

    0x000d 00013 (z.go:9)   JHI ,22

    0x000f 00015 (z.go:9)   CALL    ,runtime.morestack_noctxt(SB)

    0x0014 00020 (z.go:9)   JMP ,0

    0x0016 00022 (z.go:9)   SUBQ    $24,SP

    0x001a 00026 (z.go:10)  MOVQ    $0,AX

    0x001c 00028 (z.go:10)  CMPQ    AX,$10

    0x0020 00032 (z.go:10)  JGE $0,74

    0x0022 00034 (z.go:11)  MOVQ    AX,"".i+16(SP)

    0x0027 00039 (z.go:11)  MOVQ    AX,(SP)

    0x002b 00043 (z.go:11)  MOVQ    $"".getNumber·f+0(SB),CX

    0x0032 00050 (z.go:11)  PUSHQ   CX,

    0x0033 00051 (z.go:11)  PUSHQ   $16,

    0x0035 00053 (z.go:11)  PCDATA  $0,$0

    0x0035 00053 (z.go:11)  CALL    ,runtime.newproc(SB)

    0x003a 00058 (z.go:11)  POPQ    ,CX

    0x003b 00059 (z.go:11)  POPQ    ,CX

    0x003c 00060 (z.go:10)  MOVQ    "".i+16(SP),AX

    0x0041 00065 (z.go:10)  INCQ    ,AX

    0x0044 00068 (z.go:10)  NOP ,

    0x0044 00068 (z.go:10)  CMPQ    AX,$10

    0x0048 00072 (z.go:10)  JLT $0,34

    0x004a 00074 (z.go:13)  MOVQ    $5,(SP)

    0x0052 00082 (z.go:13)  PCDATA  $0,$0

    0x0052 00082 (z.go:13)  CALL    ,time.Sleep(SB)

    0x0057 00087 (z.go:14)  ADDQ    $24,SP

    0x005b 00091 (z.go:14)  RET ,

然而,没有办法检索这些结果。


查看完整回答
反对 回复 2021-09-10
?
素胚勾勒不出你

TA贡献1827条经验 获得超9个赞

引用Go 语言规范:Go 语句

如果函数有任何返回值,它们会在函数完成时被丢弃。

因此允许将返回值作为 goroutine 执行函数 - 它没有任何问题,并且规范明确指出它们的返回值被简单地丢弃,它不会导致任何错误,但您不会以通常的方式得到它(就像您直接调用该函数一样)。


查看完整回答
反对 回复 2021-09-10
?
忽然笑

TA贡献1806条经验 获得超5个赞

值被丢弃。并没有什么特别的go声明。你也可以写


...

_ = getNumber(i)

...

要不就


...

getNumber(i)

...

甚至


查看完整回答
反对 回复 2021-09-10
  • 3 回答
  • 0 关注
  • 292 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信