脚本之家,脚本语言编程技术及教程分享平台!
分类导航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|shell|

服务器之家 - 脚本之家 - Golang - 了解 Go 中原子操作的重要性与使用方法

了解 Go 中原子操作的重要性与使用方法

2023-11-07 19:00爱发白日梦的后端 Golang

在 Go 中,原子操作是确保并发程序正确性和性能的重要工具。通过允许对共享内存进行安全操作,它们使开发人员能够编写高效可靠的并发代码。

引言

并发是现代软件开发的一个基本方面,而在 Go 中编写并发程序相对来说是一个相对轻松的任务,这要归功于其强大的并发支持。

Go 提供了对原子操作的内置支持,这在同步并发程序中起着至关重要的作用。在本篇博客文章中,我们将探索 Go 中原子操作的概念,了解为什么它们是重要的,以及如何有效地使用它们。

什么是 Go 中的原子操作?

在 Go 中,原子操作是无需中断或受其他并发操作干扰而执行的操作。它们用于确保对共享变量的某些操作被原子地执行,这意味着它们作为一个单一的、不可分割的单元执行,并且不受其他 goroutine 或线程的干扰或数据竞争的影响。

Go 提供了一个名为 sync/atomic 的包,其中包含一组用于对原始数据类型(如整数和指针)执行原子操作的函数。在 Go 中,一些常用的原子操作包括:

  • Load(加载)
  • atomic.Load* 函数用于原子地读取变量的值。例如,atomic.LoadInt32 用于原子地加载 int32 变量的值。
  • Store(存储)
  • atomic.Store* 函数用于原子地设置变量的值。例如,atomic.StoreInt32 用于原子地设置 int32 变量的值。

  • Add 和 Subtract(增加和减少)

  • atomic.Add* 和 atomic.Sub* 函数用于原子地增加或减少变量的值。

  • Compare and Swap(CAS,比较并交换)

  • atomic.CompareAndSwap* 函数用于原子地比较变量的当前值与期望值,并在它们匹配时将变量设置为一个新值。这通常用于实现无锁的数据结构和算法。

  • Swap(交换)

  • atomic.Swap* 函数用于原子地交换变量的值与一个新值。

这些原子操作在并发环境中与共享变量一起使用时非常有价值,可以防止数据竞争,并确保对变量的操作安全且一致地执行。它们有助于构建并发数据结构、同步原语以及以线程安全的方式管理共享资源。

使用这些操作时是否需要互斥锁?

在 Go 中,sync/atomic 包提供了原子操作,可以在没有互斥锁的情况下对共享变量进行原子更新。使用原子操作的主要优势是它们通常比传统的互斥锁更高效,特别是对于像整数和指针这样的简单的原始数据类型的简单操作。

使用原子操作时不需要互斥锁,因为这些操作被设计为线程安全的,并且可以在不需要显式锁定和解锁互斥锁的情况下进行原子更新。原子操作在硬件级别上操作,确保操作的原子性,防止数据竞争,并避免传统锁定机制的需求。

然而,需要注意的是,原子操作也有其局限性。它们最适合用于对简单的、低级别的原始数据类型进行简单的更新。如果需要执行涉及多个变量或需要更复杂的同步的更复杂操作,则可能仍然需要使用互斥锁或其他同步原语。

总之,虽然原子操作可以在简单的原子更新共享变量的情况下不使用互斥锁,但是在选择原子操作和互斥锁之间取决于具体任务的需求和复杂性。根据并发代码的具体需求,选择合适的同步机制非常重要。

示例代码

package main

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

func main() {
 var counter int32

 // 创建一个 goroutine 来增加计数器的值。
 go func() {
  for i := 0; i < 5; i++ {
   atomic.AddInt32(&counter, 1)
   fmt.Printf("增加: %d\\n", atomic.LoadInt32(&counter))
   time.Sleep(time.Millisecond)
  }
 }()

 // 创建一个 goroutine 来减少计数器的值。
 go func() {
  for i := 0; i < 5; i++ {
   atomic.AddInt32(&counter, -1)
   fmt.Printf("减少: %d\\n", atomic.LoadInt32(&counter))
   time.Sleep(time.Millisecond)
  }
 }()

 // 等待 goroutine 结束。
 time.Sleep(2 * time.Second)

 fmt.Printf("最终值: %d\\n", atomic.LoadInt32(&counter))
}

运行以上示例代码,我们可以看到一个类型为 int32 的共享计数器变量。

创建了两个 goroutine,一个用于增加计数器的值,另一个用于减少计数器的值。我们使用 atomic.AddInt32 来原子地增加或减少计数器的值。我们使用 atomic.LoadInt32 来安全地加载计数器的值以供打印。程序使用 time.Sleep 等待 goroutine 结束。使用原子操作可以确保计数器在没有互斥锁的情况下安全地更新。你应该看到计数器在没有竞争的情况下正确地增加和减少。

该事件序列演示了操作的正确交错,最终计数器的值为 0。

这个输出证实了原子操作的工作方式,确保共享数据的安全性,而无需使用互斥锁进行同步。

结论

在 Go 中,原子操作是确保并发程序正确性和性能的重要工具。通过允许对共享内存进行安全操作,它们使开发人员能够编写高效可靠的并发代码。然而,在处理 Go 应用程序中的并发时,合理使用原子操作并了解潜在的权衡是非常重要的。通过对原子操作有着扎实的理解并正确使用,您可以构建健壮且响应迅速的并发程序。

 

原文地址:https://mp.weixin.qq.com/s/vQWNbkBcuRdH8rOrE4ynsQ

延伸 · 阅读

精彩推荐
  • GolangGolang爬虫框架 colly的使用

    Golang爬虫框架 colly的使用

    本文主要介绍了Golang爬虫框架 colly的使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小...

    小馨whisper7082022-07-11
  • Golanggolang为什么要统一错误处理

    golang为什么要统一错误处理

    这篇文章主要介绍了golang为什么要统一错误处理,统一错误处理的目的是为了前端开发接收到后端的statuscode,之后便于前端逻辑上开发以及开发,下文具体...

    峰啊疯了9042022-09-13
  • Golanggolang中命令行库cobra的使用方法示例

    golang中命令行库cobra的使用方法示例

    这篇文章主要给大家介绍了关于golang中命令行库cobra的使用方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需...

    wangbin5342020-05-18
  • GolangGolang通脉之数据类型详情

    Golang通脉之数据类型详情

    这篇文章主要介绍了Golang通脉之数据类型,在编程语言中标识符就是定义的具有某种意义的词,比如变量名、常量名、函数名等等,Go语言中标识符允许由...

    4032021-11-24
  • GolangGolang连接Redis数据库的方法

    Golang连接Redis数据库的方法

    这篇文章主要介绍了Golang连接Redis数据库的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...

    发现美的眼睛11892021-03-06
  • GolangGo1.18都出泛型了速来围观

    Go1.18都出泛型了速来围观

    泛型允许程序员在强类型程序设计语言中编写代码时使用一些以后才指定的类型,在实例化时作为参数指明这些类型,本文通过例子给大家介绍下如何使用...

    jiangxiaoju8572022-09-07
  • Golangweb项目中golang性能监控解析

    web项目中golang性能监控解析

    这篇文章主要为大家介绍了web项目中golang性能监控详细的解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步早日升职加薪...

    Jeff的技术栈9162022-09-15
  • GolanggRPC超时拦截器实现示例

    gRPC超时拦截器实现示例

    这篇文章主要为大家介绍了gRPC超时拦截器实现示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪...

    尹东勋4602022-10-20