摘要:
如果f2是独占的,您可以用这种方式修改它,它将在{time.Sleep fmt.Println}}funcf1{for{*x=*x+1 time.Sleep}}funcf2{for{*y=*y+1}}smokezl·#2·1 floor之前更改packagemainimportvarx,yintfuncmain(){fmt.Printlnruntime.GOMAXPROCSgo1(&x)gof2(&y)。我认为应该是这样的。如果f1和f2不休眠,则相当于独占。占用了M,我没有机会写入主内存我也测试了它。如果它是地址,则值确实会被修改。感谢上帝~TOMFATCAT·#3·7月之前,这完全是为了安全。如果要更改全局变量,可以将其放入chan中。你应该在交流中分享记忆。大致可以理解,否则,所有协程将一起写入相同的内存。很容易编写packagemainimportvarx而无需锁定,yintfuncmain(){fmt.Printlnruntime.GOMAXPROCSx=100y=33gof1(&x)gof2(&y)for{time.Sleepfmt.Println}}funcf1{for{*x=*x+1 time.Sleep}}}fungf2{for{*y=*y+1}

go修改全局变量的问题

测试 goroutine 修改全局变量,有x y 两个全局函数,分别在两个 goroutine f1() 和 f2() 中修改(f1()修改x,f2()修改y),f2() 只多了一行sleep,为什么最终主 goroutine 打印y时,y的值一直是0?? 代码如下

package main

import (
    "fmt"
    "runtime"
    "time"
)

var x, y int

func main() {
    runtime.GOMAXPROCS(3)
    go f1()
    go f2()
    for {
        time.Sleep(time.Second * 1)
        fmt.Println("x:", x, "y", y)
    }
}

func f1() {
    for {
        x = x + 1
        time.Sleep(time.Second * 1)
    }
}

func f2() {
    for {
        y = y + 1
    }
}

  

结果:

goroutine 修改全局变量无效问题第1张

package main

import (
    "fmt"
    "runtime"
    "time"
)

var x, y int

func main() {
    fmt.Println(runtime.NumCPU())
    runtime.GOMAXPROCS(3)
    go f1(&x)
    go f2(&y)
    for {
        time.Sleep(time.Second * 1)
        fmt.Println("x:", x, "y", y)
    }
}

func f1(x *int) {
        for {
                *x = *x + 1
                time.Sleep(time.Second * 1)
        }
}

func f2(y *int) {
        for {
                *y = *y + 1
        }
}
smokezl
smokezl · #2 · 7月之前

1楼 @darren3126 对的,我觉得应该是这样,f1 和 f2 如果都不sleep 的话,相当于分别都独占了M,没有机会写主存 </br>我也测试过,如果是传址的话,值确实会被修改,感谢大神~

TOMFATCAT
TOMFATCAT · #3 · 7月之前

这完全是为了安全,如果你想改变全局变量,可以把它塞到chan里面,要在通信里共享内存,大概是这么理解的,不然所有协程一块去写同一块内存,不加锁还,很容易写花的

package main

import (
    "fmt"
    "runtime"
    "time"
)

var x, y int

func main() {
    fmt.Println(runtime.NumCPU())
    runtime.GOMAXPROCS(3)
    x = 100
    y = 33
    go f1(&x)
    go f2(&y)
    for {
        time.Sleep(time.Second * 1)
        fmt.Println("x:", x, "y", y)
    }
}

func f1(x *int) {
        for {
                *x = *x + 1
                time.Sleep(time.Second * 1)
        }
}

func f2(y *int) {
        for {
                *y = *y + 1
        }
}