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

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

服务器之家 - 脚本之家 - Golang - 揭秘 Go 中 Goroutines 轻量级并发

揭秘 Go 中 Goroutines 轻量级并发

2023-12-23 14:45技术的游戏 Golang

本文将全面概述 Goroutines,它们的轻量级特性,如何使用 go 关键字创建它们,以及它们提出的同步挑战,包括竞态条件和共享数据问题。

并发是现代软件开发的一个基本概念,使程序能够同时执行多个任务。在 Go 编程领域,理解 Goroutines 是至关重要的。本文将全面概述 Goroutines,它们的轻量级特性,如何使用 go 关键字创建它们,以及它们提出的同步挑战,包括竞态条件和共享数据问题。

揭秘 Go 中 Goroutines 轻量级并发

Goroutines 解释

Goroutine 是 Go 编程语言中并发编程的基本构建块。它本质上是一个轻量级的执行线程,可以与 Go 程序中的其他 Goroutines 同时并发运行。与其他编程语言中的传统线程不同,Goroutines 由 Go 运行时管理,并且在内存和 CPU 利用率方面更加高效。

轻量级特性与效率

Goroutines 的一个显著特点是它们的 轻量级 特性。传统的线程可能会消耗大量的内存和 CPU 资源。相比之下,Goroutines 非常高效,允许您创建成千上万个而不会造成显著的开销。

Goroutines 的效率源于它们能够在较少数量的操作系统线程上进行多路复用,并根据工作负载动态调整其分配。这意味着 Go 程序可以有效地利用多个核心和处理器,无需进行大量的手动线程管理。

创建 Goroutines(使用 go 关键字)

在 Go 中创建 Goroutine 非常简单,这要归功于 go 关键字。当您在函数调用前加上 go 时,Go 会创建一个新的 Goroutine 来并发执行该函数。

package main

import (
    "fmt"
    "time"
)

func sayHello() {
    for i := 0; i < 5; i++ {
        fmt.Println("Hello, World!")
        time.Sleep(time.Millisecond * 500)
    }
}

func main() {
    go sayHello() // Start a new Goroutine
    time.Sleep(time.Second * 2)
    fmt.Println("Main function")
}

在上面的示例中,sayHello 函数与 main 函数并发执行,这使得它成为在 Go 中利用并发的一种简单而有效的方式。

同步挑战

虽然 Goroutines 在并发编程中提供了许多优势,但它们也带来了必须仔细管理的同步挑战:

1.Go 中的竞态条件

(1) 什么是竞态条件?

在 Go 程序中,当多个 Goroutines(轻量级线程)并发访问共享数据,并且至少有一个修改了数据时,就会发生 竞态条件。竞态条件会导致结果不可预测,因为执行的顺序不能保证。它们可能导致数据损坏、崩溃或不正确的程序行为。

(2) 竞态条件的示例

package main

import (
    "fmt"
    "sync"
)

var sharedCounter int
var wg sync.WaitGroup

func increment() {
    for i := 0; i < 10000; i++ {
        sharedCounter++
    }
    wg.Done()
}

func main() {
    wg.Add(2)
    go increment()
    go increment()
    wg.Wait()
    fmt.Println("Shared Counter:", sharedCounter)
}

在这个示例中,两个 Goroutines 同时增加 sharedCounter 变量而没有同步。这可能会导致竞态条件,其中 sharedCounter 的最终值是不可预测的,且很可能是不正确的。

(3) 缓解竞态条件

为了在 Go 中缓解竞态条件,您可以使用同步原语,如互斥锁(Mutex,即 mutual exclusion locks)。互斥锁确保一次只有一个 Goroutine 可以访问代码的关键部分。以下是使用互斥锁进行适当同步的先前示例的更新版本:

package main

import (
    "fmt"
    "sync"
)

var sharedCounter int
var wg sync.WaitGroup
var mu sync.Mutex

func increment() {
    for i := 0; i < 10000; i++ {
        mu.Lock()
        sharedCounter++
        mu.Unlock()
    }
    wg.Done()
}

func main() {
    wg.Add(2)
    go increment()
    go increment()
    wg.Wait()
    fmt.Println("Shared Counter:", sharedCounter)
}

在这个修订后的代码中,我们使用 mu 互斥锁来保护修改 sharedCounter 的关键代码段。通过锁定和解锁互斥锁,我们确保一次只有一个 Goroutine 可以访问和修改 sharedCounter,从而消除了竞态条件。

2.Go 中的共享数据问题

(1) 理解共享数据问题

在 Go 中,当多个 Goroutines 在没有适当同步的情况下同时访问和操作共享数据时,就会出现共享数据问题。这些问题主要以两种形式出现:

  • 数据竞态(Data Races): 当两个或更多 Goroutines 同时访问共享数据时,可能会导致不可预测的结果。数据竞态可能导致数据损坏或程序行为不正确。
  • 死锁(Deadlocks): 当 Goroutines 互相等待释放资源时,可能会发生死锁。这可能导致程序停滞不前。

(2) 缓解共享数据问题

为了在 Go 中缓解共享数据问题,开发者应该使用适当的同步机制,如互斥锁、通道和其他同步原语。以下是一些最佳实践:

  • 使用互斥锁:使用互斥锁来保护共享数据,确保一次只有一个 Goroutine 可以访问它。
  • 使用通道:通道为 Goroutines 提供了一种安全的方式来通信和共享数据。它们通过确保对共享数据的控制访问来帮助防止数据竞态。
  • 避免循环依赖:在创建 Goroutines 互相等待释放资源(从而导致死锁)的情况下,要谨慎。仔细的设计可以帮助您避免这种情况。

总之,在 Go 中编写并发程序时,管理竞态条件和共享数据问题至关重要。通过了解这些问题并实施适当的同步技术,开发者可以创建出充分利用 Go 并发支持的健壮可靠的并发应用,同时避免与共享数据操作相关的陷阱。

总的来说,Goroutines 是 Go 编程语言的一个强大特性,提供了一种轻量级和高效的并发实现方式。通过使用 go 关键字,开发者可以轻松创建 Goroutines 来并发执行任务。然而,在构建 Go 中的并发应用时,了解诸如竞态条件和共享数据问题等同步挑战,并采用适当的技术来解决它们,是非常关键的。

原文地址:https://mp.weixin.qq.com/s?__biz=MzUzMTY1MDAwMQ==&mid=2247486827&idx=1&sn=dfdbdcf44aa0d86aae45cd1e2f96c3f4

延伸 · 阅读

精彩推荐
  • GolangGo 在 MongoDB 中常用查询与修改的操作

    Go 在 MongoDB 中常用查询与修改的操作

    这篇文章主要介绍了Go 在 MongoDB 中常用查询与修改的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    Smicry9622021-06-23
  • GolangGolang 获取系统信息的实现

    Golang 获取系统信息的实现

    本文主要介绍了Golang 获取系统信息的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小...

    李迟7942022-10-12
  • Golang详解Golang开启http服务的三种方式

    详解Golang开启http服务的三种方式

    这篇文章主要介绍了详解Golang开启http服务的三种方式,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们...

    L千年老妖35672020-07-21
  • GolangGo语言判断文件或文件夹是否存在的方法

    Go语言判断文件或文件夹是否存在的方法

    这篇文章主要介绍了Go语言判断文件或文件夹是否存在的方法,结合具体实例形式对比分析了Go语言针对文件与目录判断的操作技巧与相关注意事项,需要的朋...

    phpgo5842020-05-06
  • GolangGo语言实现一个简单的并发聊天室的项目实战

    Go语言实现一个简单的并发聊天室的项目实战

    本文主要介绍了Go语言实现一个简单的并发聊天室的项目实战,文中根据实例编码详细介绍的十分详尽,具有一定的参考价值,感兴趣的小伙伴们可以参考...

    Sirius_74442022-09-07
  • Golanggolang数据结构之golang稀疏数组sparsearray详解

    golang数据结构之golang稀疏数组sparsearray详解

    这篇文章主要介绍了golang数据结构之golang稀疏数组sparsearray的相关知识,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要...

    墨儿舞步4252021-09-18
  • GolangGolang 数据库操作(sqlx)和不定字段结果查询

    Golang 数据库操作(sqlx)和不定字段结果查询

    本文主要介绍了Golang 数据库操作(sqlx)和不定字段结果查询,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    小小小丶叶子10332021-09-18
  • GolangGolang学习笔记之延迟函数(defer)的使用小结

    Golang学习笔记之延迟函数(defer)的使用小结

    这篇文章主要介绍了Golang学习笔记之延迟函数(defer),小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧 ...

    学生黄哲4862020-05-21