MST
星途 面试题库

面试题:Go中原子操作在并发控制中的基本应用

请描述在Go语言中,如何使用原子操作(如`atomic`包中的函数)来实现简单的并发计数器?并说明这样做相较于普通变量自增的优势。
28.2万 热度难度
编程语言Go

知识考点

AI 面试

面试题答案

一键面试
  1. 使用原子操作实现并发计数器: 在Go语言中,可以通过atomic包中的AddInt64函数来实现并发计数器。示例代码如下:
package main

import (
    "fmt"
    "sync"
    "sync/atomic"
)

func main() {
    var counter int64
    var wg sync.WaitGroup
    numRoutines := 10

    for i := 0; i < numRoutines; i++ {
        wg.Add(1)
        go func() {
            defer wg.Done()
            atomic.AddInt64(&counter, 1)
        }()
    }

    wg.Wait()
    fmt.Println("Final counter value:", counter)
}

在上述代码中:

  • 首先声明了一个int64类型的counter变量作为计数器。
  • 使用sync.WaitGroup来等待所有的goroutine完成。
  • 在每个goroutine中,通过atomic.AddInt64(&counter, 1)counter进行原子自增操作。AddInt64函数的第一个参数是要操作的变量的指针,第二个参数是要增加的值。
  1. 相较于普通变量自增的优势
    • 线程安全:在并发环境下,普通变量自增操作不是原子的,会出现竞态条件(race condition)。例如,当多个goroutine同时对一个普通变量进行自增操作时,可能会导致结果不准确。而使用原子操作,atomic包中的函数保证了操作的原子性,多个goroutine同时操作时不会出现竞态条件,确保了数据的一致性和正确性。
    • 无需额外同步机制:使用普通变量自增时,为了保证线程安全,通常需要使用锁(如sync.Mutex)来保护对变量的操作。而原子操作本身就是线程安全的,不需要额外的锁机制,这样可以减少锁竞争带来的开销,提高并发性能。在高并发场景下,锁的争用可能会成为性能瓶颈,原子操作能有效避免这种情况。