登录
首页 >  Golang >  Go问答

在 Go 中并发读取/访问数组是安全的吗?

来源:stackoverflow

时间:2024-03-09 15:18:25 175浏览 收藏

偷偷努力,悄无声息地变强,然后惊艳所有人!哈哈,小伙伴们又来学习啦~今天我将给大家介绍《在 Go 中并发读取/访问数组是安全的吗?》,这篇文章主要会讲到等等知识点,不知道大家对其都有多少了解,下面我们就一起来看一吧!当然,非常希望大家能多多评论,给出合理的建议,我们一起学习,一起进步!

问题内容

就像我有一个带有数组的结构,并且我想做这样的事情

type Paxos struct {
    peers      []string
}

for _, peer := range px.peers {
   \\do stuff
}

我的例程/线程永远不会修改对等数组,只是从中读取。 peers是一个服务器地址数组,服务器可能会失败,但这不会影响peers数组(后面的rpc调用只会失败)


正确答案


如果不涉及写入,则无论数据结构如何,并发读取始终是安全的。然而,只要涉及到对变量的单个并发不安全写入,您就需要串行化对该变量的并发访问(写入和读取)。

此外,在不超过一个 goroutine 写入给定元素的情况下,您可以安全地写入切片或数组的元素。

例如,如果您在竞争检测器打开的情况下运行以下 programme,则可能会报告竞争条件,因为多个 goroutine 同时修改变量 results ,而没有采取任何预防措施:

package main

import (
    "fmt"
    "sync"
)

func main() {
    const n = 8
    var results []int
    var wg sync.waitgroup
    wg.add(n)
    for i := 0; i < n; i++ {
        i := i
        go func() {
            defer wg.done()
            results = append(results, square(i))
        }()
    }
    wg.wait()
    fmt.println(results)

}

func square(i int) int {
    return i * i
}

但是,下面的programme不包含这样的不同步错误,因为切片的每个元素都是由单个goroutine修改的:

package main

import (
    "fmt"
    "sync"
)

func main() {
    const n = 8
    results := make([]int, n)
    var wg sync.WaitGroup
    wg.Add(n)
    for i := 0; i < n; i++ {
        i := i
        go func() {
            defer wg.Done()
            results[i] = square(i)
        }()
    }
    wg.Wait()
    fmt.Println(results)

}

func square(i int) int {
    return i * i
}

以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于Golang的相关知识,也可关注golang学习网公众号。

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>