美文网首页
Golang并发操作变量需要注意的问题

Golang并发操作变量需要注意的问题

作者: bysir | 来源:发表于2019-01-15 17:08 被阅读0次

思考源于这个问题: golang 中并发读写同一个变量会出现部分错乱吗?

要记得golang中变量的赋值不是并发安全的

什么是并发安全

我理解的并发安全就是当并发和不并发的情况下执行结果是一致的.

比如 count++, ok = !ok, 在非并发下很好理解, 而在并发情况下却容易出现预期之外的结果, 这样的代码就是非并发安全的.

举个例子:
count++ 其实是分成两步执行的. 当分成了两步执行, 那么其他协程就可以趁着这个时间间隙作怪.

如一下 ab两个协程同时 count++

count:= 1
a > 读取count : 1
b > 读取count : 1
a > 计算count+1 : 2
b > 计算count+1 : 2
a > 赋值count : 2
b > 赋值count : 2

这就会发生明明ab协程计算了两次, 可结果还是2.

赋值一个简单的count都会出现偏差, 那么赋值一个更为复杂的结构体会不会有问题呢?

例如以下代码, 会进入x.Y != x.X判断分支(概率低 但总会发生). 如果还有其他协程再去读x变量, 则会引发逻辑错误.


func TestX(t *testing.T) {
    x := struct {
        X string
        Y string
    }{}

    for i := 0; i < 300000; i++ {
        go func() {
            y := strconv.FormatInt(int64(i), 10)
            x = struct {
                X string
                Y string
            }{
                X: y,
                Y: y,
            }
            if x.Y != x.X {
                t.Log("-----", x)
            }
        }()
    }

    time.Sleep(1 * time.Second)

    t.Log(x)
}

可以想到,
在结构体中有多个字段, a协程赋值了一些字段(x字段), b协程赋值了一些字段(y字段), 此时的整个结构体既不是a协程想要的数据, 也不是b协程想要的数据.

如何解决这个问题呢: 使用atomic.Value

func TestY(t *testing.T) {
    v := atomic.Value{}
    for i := 0; i < 300000; i++ {
        go func() {
            y := strconv.FormatInt(int64(i), 10)

            v.Store(struct {
                X string
                Y string
            }{
                X: y,
                Y: y,
            })

            x := v.Load().(struct {
                X string
                Y string
            })
            if x.Y != x.X {
                t.Log("-----", x)
            }
        }()
    }

    time.Sleep(1 * time.Second)

    t.Log(v.Load())
}

这时候就不会进入x.Y != x.X分支了.

结语

在编写并发代码的时候一定记得处理这些问题(即便只是一个基本类型的变量): 可以通过加锁, 或者不使用并发读写的方式(使用channel 或者 函数式编程).

相关文章

  • Golang并发操作变量需要注意的问题

    思考源于这个问题: golang 中并发读写同一个变量会出现部分错乱吗? 要记得golang中变量的赋值不是并发安...

  • Go append并发问题

    Golang append并发问题

  • golang语言map的并发和排序

    golang语言map的并发和排序 map的并发问题 golang缺省的map不是thread safe的,如果存...

  • golang进行交叉编译

    golang进行交叉编译 交叉编译即编译不同操作系统平台的可执行程序 golang执行交叉编译,只需要使用两个变量...

  • 线程本地存储模式

    多个线程同时读写同一共享变量存在并发问题。要解决这个问题,我们可以突破的写,没有写操作自然没有并发问题了。其实还可...

  • 并发map --- sync map分析

    [TOC] 本文基于1.10源码分析如之前的文章可以看到,golang中的map是不支持并发操作的,golang推...

  • Golang之并发编程一

    并发基础 在说Golang的并发编程之前,先认识一下目前并发的几种实现方式: 1.多进程。操作系统实现的并发模型,...

  • Rust线程间传递字符串与并发编程

    问题背景 在并发编程场景下,消息传递是一个最基本的操作,例如,在Golang中,可以简单的通过chan来传递,而在...

  • golang 指针

    1.Golang提供了指针用于操作数据内存,并通过引用来修改变量。只声明未赋值的变量,golang都会自动为其初始...

  • volatile 关键字

    注意:volatile 不是原子性的,所以不能保证并发问题如:volatile 修饰一个变量 i = ,0,开辟...

网友评论

      本文标题:Golang并发操作变量需要注意的问题

      本文链接:https://www.haomeiwen.com/subject/nmnicqtx.html