为什么golang没有volatile?

缘起
从 java 转 golang 程序员可能会有一个疑问:为什么 golang 没有 volatile 关键字?这样的话如何保证可见性?

在 java 中我们一般可以通过给某个变量加上 volatile 修饰符来保证不同线程对该变量读写时的可见性,并凭借 JVM 的 happens-before 机制避免了 data race,关于 data race 和 happens-before 机制,这里不做过多阐述,有疑问者可自行 google。

正是由于上述 JVM 全方位保姆式的完善保护机制,java 程序员们习以为常地写出无锁代码:

publicclass Connection{
privatestaticvolatile Boolean initialized = false;
public void init(){
//init process…
initialized = true;
}
public Boolean isInitialized(){
return initialized;
}
}
上述代码可以在一个线程中调用 init,然后在其他线程中循环调用 isInitialized 去检测初始化是否完成,不会产生任何可见性问题和 data race。

但是如果使用 golang 写个类似的例子试试:

var inited = false

func Setup(){
time.Sleep(time.Second)
inited = true
}
func main() {
go Setup()

for{
	if inited{
		break
	}
	time.Sleep(100 * time.Millisecond)
}

fmt.Println("setup succeed")

}
你可以使用 go run -race 运行上边的程序,你会看到警告:

➜ go-learning git:(master) ✗ go run -race main/data_race.go

WARNING: DATA RACE
Write at 0x0000012272c1 by goroutine 7:
main.Setup()
/Users/yalouwang/workspace/go/go-learning/main/data_race.go:20 +0x47

Previous read at 0x0000012272c1 by main goroutine:
main.main()
/Users/yalouwang/workspace/go/go-learning/main/data_race.go:26 +0x65

Goroutine 7 (running) created at:
main.main()
/Users/yalouwang/workspace/go/go-learning/main/data_race.go:23 +0x46

setup succeed
Found 1 data race(s)
exit status 66
-race 用于检测代码中可能存在的 data race。这里的告警信息写得很清楚,一个 goroutine 的写操作和另一个 goroutine 的读操作可能发生 data race,有风险。

你检查完代码后发现,应该在 inited 前加一个"volatile";但是刚刚想要改代码,却发现 golang 中根本没有这个关键字!

于是你大声问道:golang 的 volatile 去哪儿了?

解决问题
先看看如何解决上边的 data race 问题。实际上在 golang 中,互斥锁在并发操作中是非常常见的,当某个变量存在并发访问的可能时,请一定记得加锁:

var inited = false
var lock sync.Mutex

func Setup(){
time.Sleep(time.Second)
lock.Lock()
inited = true
lock.Unlock()
}

func main() {
go Setup()

for{
	lock.Lock()
	b := inited
	lock.Unlock()
	if b{
		break
	}
	time.Sleep(100 * time.Millisecond)
}

fmt.Println("setup succeed")

}
再次运行 go run -race,我们发现不再告警了。

好了,解决了旧问题,新的问题又来了:加锁不就降低并发性能了吗?

新的理念
在 c/c++以及 java 中,我们往往习惯设置共享对象或基本类型,然后在不同的线程中对它进行读写操作从而达到线程间通信的目的。这样做的好处就是操作简单,坏处也很明显:容易造成 data race,而为了避免 data race 又不得不加锁,加锁又降低了并发性能。

所以 golang 提出了一个新的理念:

Do not communicate by sharing memory; instead, share memory by communicating.

——出自 effective go

乍一看有点玄乎,但实际上它表达的意思很简单:别再用共享变量啦,快来试试偶们的 channel 吧!

关于 channel 的使用相信大家已经很熟了,我们用 chan 改造一下上边的代码:

func Setup2() <-chan bool {
time.Sleep(time.Second * 3)
c := make(chanbool)
c <- true
return c
}

func main() {
if <-Setup2(){
fmt.Println(“setup succeed”)
}
}
我们得到了一个无锁的版本,是不是十分简单且优雅呢?事实上使用 chan 的好处不仅在于解决了并发访问的 data race 和锁的问题,而且还提高了代码的运行效率——chan 的接收端 goroutine 会被挂起 直到 chan 中有值可读,相比于传统方法的循环检测共享变量,这种方式效率明显要高不少,且优雅。

所以,volatile 呢?
相信大多数读者看到这里,对开头的问题已经有答案了。

volatile 呢?他早就被被 golang 的创造者们扔进了历史的垃圾桶啦!放弃它和它老旧的共享变量思想吧,是时候使用 channel 重构你的代码了!

服务推荐

  • 蜻蜓代理
  • 代理ip
  • 微信域名拦截检测
  • 微信域名检测api

你可能感兴趣的:(为什么golang没有volatile?)