Lectura del índice
La forma principal de administrar el estado en Go es a través de la comunicación entre canales, que encontramos en el ejemplo del grupo de subprocesos, pero hay otras formas de administrar el estado.
Aquí veremos cómo sync/atomic
realizar un conteo atómico en múltiples rutinas de Go usando el paquete.
// atomic-counters.go
package main
import (
"fmt"
"runtime"
"sync/atomic"
"time"
)
func main() {
// 我们将使用一个无符号整型数来表示(永远是正整数)这个计数器。
var ops uint64 = 0
// 为了模拟并发更新,我们启动 50 个 Go 协程,对计数
// 器每隔 1ms (译者注:应为非准确时间)进行一次加一操作。
for i := 0; i < 50; i++ {
go func() {
for {
// 使用 `AddUint64` 来让计数器自动增加,使用
// `&` 语法来给出 `ops` 的内存地址。
atomic.AddUint64(&ops, 1)
// 允许其它 Go 协程的执行
runtime.Gosched()
}
}()
}
// 等待一秒,让 ops 的自加操作执行一会。
time.Sleep(time.Second)
// 为了在计数器还在被其它 Go 协程更新时,安全的使用它,
// 我们通过 `LoadUint64` 将当前值的拷贝提取到 `opsFinal`
// 中。和上面一样,我们需要给这个函数所取值的内存地址 `&ops`
opsFinal := atomic.LoadUint64(&ops)
fmt.Println("ops:", opsFinal)
}
[root@bogon test]# go run atomic-counters.go
ops: 6171555
[root@bogon test]#