摘要:
如果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 } }
结果:
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 · #2 · 7月之前
1楼 @darren3126 对的,我觉得应该是这样,f1 和 f2 如果都不sleep 的话,相当于分别都独占了M,没有机会写主存 </br>我也测试过,如果是传址的话,值确实会被修改,感谢大神~
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 } }