当前位置:首页 > 科技  > 软件

使用互斥锁(Mutex)管理共享资源

来源: 责编: 时间:2023-12-25 09:35:47 391观看
导读在Go中确保并发安全性并发是Go中的一个强大功能,它允许多个Goroutines(并发线程)同时执行。然而,伴随着强大的功能也带来了大量的责任。当多个Goroutines并发地访问和修改共享资源时,可能会导致数据损坏、数据竞争(race con

在Go中确保并发安全性

并发是Go中的一个强大功能,它允许多个Goroutines(并发线程)同时执行。然而,伴随着强大的功能也带来了大量的责任。当多个Goroutines并发地访问和修改共享资源时,可能会导致数据损坏、数据竞争(race conditions)和不可预测的程序行为。为了解决这些问题,Go提供了一种称为互斥锁(Mutex,互斥排他锁的缩写)的同步原语。在本文中,我们将探讨互斥锁在管理共享资源中的作用,以及在并发编程中使用它的必要性。6H028资讯网——每日最新资讯28at.com

6H028资讯网——每日最新资讯28at.com

互斥锁简介

互斥锁是一种同步原语,用于提供对共享资源或代码关键部分的独占访问。它充当了门卫的角色,一次只允许一个Goroutine访问和修改受保护的资源。当一个Goroutine持有互斥锁时,所有试图获取它的其他Goroutines都必须等待。6H028资讯网——每日最新资讯28at.com

互斥锁提供了两个基本方法:6H028资讯网——每日最新资讯28at.com

  • Lock(): 这个方法获取互斥锁,授予对资源的独占访问。如果另一个Goroutine已经持有该互斥锁,新的Goroutine将被阻塞,直到它被释放。
  • Unlock(): 这个方法释放互斥锁,允许其他等待的Goroutines获取它并访问资源。

互斥锁的必要性

使用互斥锁的原因在于,当多个Goroutines并发访问共享资源时,这些资源容易遭受数据竞争和不一致性的风险。以下是互斥锁至关重要的一些常见场景:6H028资讯网——每日最新资讯28at.com

1. 数据竞争

数据竞争发生在多个Goroutines并发访问共享数据时,其中至少一个Goroutine对其进行修改。这可能导致不可预测和错误的行为,因为执行顺序是不确定的。互斥锁通过一次只允许一个Goroutine访问共享资源来帮助防止数据竞争。6H028资讯网——每日最新资讯28at.com

package mainimport (    "fmt"    "sync")var sharedData intvar mu sync.Mutexfunc increment() {    mu.Lock()    sharedData++    mu.Unlock()}func main() {    var wg sync.WaitGroup    for i := 0; i < 100; i++ {        wg.Add(1)        go func() {            defer wg.Done()            increment()        }()    }    wg.Wait()    fmt.Println("Shared Data:", sharedData)}

在这个示例中,多个Goroutines并发地增加sharedData变量,如果没有使用互斥锁,这将导致数据竞争。6H028资讯网——每日最新资讯28at.com

2. 临界区(Critical Sections)

临界区是访问共享资源的代码部分。当多个Goroutines试图同时访问同一个临界区时,可能会导致不可预测的行为。互斥锁确保一次只有一个Goroutine进入临界区,从而保证对共享资源的有序访问。6H028资讯网——每日最新资讯28at.com

package mainimport (    "fmt"    "sync")var (    sharedResource int    mu             sync.Mutex)func updateSharedResource() {    mu.Lock()    // Critical section: Access and modify sharedResource    sharedResource++    mu.Unlock()}func main() {    var wg sync.WaitGroup    for i := 0; i < 100; i++ {        wg.Add(1)        go func() {            defer wg.Done()            updateSharedResource()        }()    }    wg.Wait()    fmt.Println("Shared Resource:", sharedResource)}

在这个示例中,updateSharedResource 函数代表一个临界区,其中访问并修改了 sharedResource。如果没有使用互斥锁,对这个临界区的并发访问可能会导致不正确的结果。6H028资讯网——每日最新资讯28at.com

互斥锁定

互斥锁提供了两个基本操作:锁定和解锁。让我们首先了解互斥锁的锁定操作:6H028资讯网——每日最新资讯28at.com

锁定互斥锁:当一个Goroutine想要访问共享资源或一个临界区时,它会调用互斥锁上的Lock()方法。如果互斥锁当前是未锁定的,它将变为锁定状态,从而允许Goroutine继续执行。如果互斥锁已被另一个Goroutine锁定,调用的Goroutine将被阻塞,直到互斥锁变为可用状态。6H028资讯网——每日最新资讯28at.com

下面是一个演示互斥锁锁定的代码示例:6H028资讯网——每日最新资讯28at.com

package mainimport (    "fmt"    "sync")func main() {    var mu sync.Mutex    mu.Lock() // Lock the Mutex    // Critical section: Access and modify shared resource    fmt.Println("Locked the Mutex")    mu.Unlock() // Unlock the Mutex}

在这个示例中,mu.Lock() 调用锁定了互斥锁,确保一次只有一个Goroutine可以进入临界区。当完成临界区后,使用 mu.Unlock() 解锁互斥锁。6H028资讯网——每日最新资讯28at.com

互斥锁解锁

解锁互斥锁:当一个Goroutine完成其临界区的执行并且不再需要对共享资源进行独占访问时,它会在互斥锁上调用 Unlock() 方法。这个操作会释放互斥锁,从而允许其他Goroutines获取它。6H028资讯网——每日最新资讯28at.com

以下是互斥锁解锁的执行方式:6H028资讯网——每日最新资讯28at.com

package mainimport (    "fmt"    "sync")func main() {    var mu sync.Mutex    mu.Lock() // Lock the Mutex    // Critical section: Access and modify shared resource    fmt.Println("Locked the Mutex")    mu.Unlock() // Unlock the Mutex    fmt.Println("Unlocked the Mutex")}

在这个示例中,在临界区之后调用了 mu.Unlock() 以释放互斥锁,使其可供其他Goroutines使用。6H028资讯网——每日最新资讯28at.com

避免死锁

尽管互斥锁是确保并发安全性的强大工具,但如果使用不当,它们也可能引入死锁。死锁 是指两个或多个Goroutines被卡住,彼此等待释放资源的情况。为了避免死锁,请遵循以下最佳实践:6H028资讯网——每日最新资讯28at.com

  • 始终解锁:确保在锁定后解锁互斥锁。如果不这样做,可能会导致死锁。
  • **使用 defer**:为了确保互斥锁始终被解锁,考虑使用 defer 语句在函数结束时解锁它们。
  • 避免循环依赖:小心循环依赖的情况,其中多个Goroutines互相等待释放资源。设计代码时要避免这种情况。
package mainimport (    "fmt"    "sync")func main() {    var mu sync.Mutex    mu.Lock() // Lock the Mutex    // Critical section: Access and modify shared resource    // Oops! Forgot to unlock the Mutex    // mu.Unlock() // Uncomment this line to avoid deadlock    fmt.Println("Locked the Mutex")    // ... Some more code    // Potential deadlock if mu.Unlock() is not called}

在这个示例中,如果遗忘或注释掉 mu.Unlock() 这一行,由于互斥锁持续保持锁定状态,可能会发生死锁。6H028资讯网——每日最新资讯28at.com

临界区

什么是临界区?6H028资讯网——每日最新资讯28at.com

在并发编程中,临界区 是指访问共享资源或变量的代码部分。它被称为“临界”是因为在任何给定时刻只应允许一个Goroutine执行它。当多个Goroutines并发访问一个临界区时,可能会导致数据损坏或竞态条件,其中执行的顺序变得不可预测。6H028资讯网——每日最新资讯28at.com

使用互斥锁保护临界区

互斥锁用于保护临界区,确保一次只有一个Goroutine可以访问它们。互斥锁提供了两个基本方法:6H028资讯网——每日最新资讯28at.com

  • Lock(): 此方法锁定互斥锁,允许当前的Goroutine进入临界区。如果另一个Goroutine已经锁定了互斥锁,调用该方法的Goroutine将被阻塞,直到互斥锁被释放。
  • Unlock(): 此方法解锁互斥锁,允许其他Goroutines获取它并进入临界区。

以下是一个演示使用互斥锁保护临界区的示例:6H028资讯网——每日最新资讯28at.com

package mainimport (    "fmt"    "sync")var sharedResource intvar mu sync.Mutexfunc updateSharedResource() {    mu.Lock() // Lock the Mutex    // Critical section: Access and modify sharedResource    sharedResource++    mu.Unlock() // Unlock the Mutex}func main() {    var wg sync.WaitGroup    for i := 0; i < 100; i++ {        wg.Add(1)        go func() {            defer wg.Done()            updateSharedResource()        }()    }    wg.Wait()    fmt.Println("Shared Resource:", sharedResource)}

在这个示例中,updateSharedResource 函数代表一个临界区,其中 sharedResource 被访问和修改。互斥锁 mu 确保一次只有一个Goroutine可以进入这个临界区。6H028资讯网——每日最新资讯28at.com

互斥锁与通道的比较

互斥锁并不是Go中管理并发的唯一工具;通道也是另一个重要的机制。以下是互斥锁和通道的简要比较:6H028资讯网——每日最新资讯28at.com

  • 互斥锁 用于保护临界区并确保对共享资源的独占访问。当您需要对数据访问进行细粒度的控制时,它们非常适用。
  • 通道 用于Goroutines之间的通信和同步。它们为交换数据和同步Goroutines提供了更高级别的抽象。

选择使用互斥锁还是通道取决于您程序的具体需求。当您需要保护共享数据时,互斥锁是理想的选择,而当通信和Goroutines之间的协调是主要关注点时,通道则表现出色。6H028资讯网——每日最新资讯28at.com

总之,互斥锁是Go中确保安全并发的强大工具。它们有助于保护临界区,防止数据竞态,并确保共享资源的完整性。理解何时以及如何使用互斥锁对于编写既高效又可靠的并发Go程序至关重要。6H028资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-53352-0.html使用互斥锁(Mutex)管理共享资源

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 拒绝写重复代码,试试这套开源的 SpringBoot 组件,效率翻倍

下一篇: 户外用电无忧,还得看正浩

标签:
  • 热门焦点
  • iPhone卖不动了!苹果股价创年内最大日跌幅:市值一夜蒸发万亿元

    8月5日消息,今天凌晨美股三大指数高开低走集体收跌,道指跌0.41%;纳指跌0.36%;标普500指数跌0.52%。热门科技股也都变化极大,其中苹果报181.99美元,跌4.8%,创
  • 三言两语说透设计模式的艺术-单例模式

    写在前面单例模式是一种常用的软件设计模式,它所创建的对象只有一个实例,且该实例易于被外界访问。单例对象由于只有一个实例,所以它可以方便地被系统中的其他对象共享,从而减少
  • 企业采用CRM系统的11个好处

    客户关系管理(CRM)软件可以为企业提供很多的好处,从客户保留到提高生产力。  CRM软件用于企业收集客户互动,以改善客户体验和满意度。  CRM软件市场规模如今超过580
  • 三万字盘点 Spring 九大核心基础功能

    大家好,我是三友~~今天来跟大家聊一聊Spring的9大核心基础功能。话不多说,先上目录:图片友情提示,本文过长,建议收藏,嘿嘿嘿!一、资源管理资源管理是Spring的一个核心的基础功能,不
  • 一文搞定Java NIO,以及各种奇葩流

    大家好,我是哪吒。很多朋友问我,如何才能学好IO流,对各种流的概念,云里雾里的,不求甚解。用到的时候,现百度,功能虽然实现了,但是为什么用这个?不知道。更别说效率问题了~下次再遇到,
  • 一条抖音4亿人围观 ! 这家MCN比无忧传媒还野

    作者:Hiu 来源:互联网品牌官01 擦边少女空降热搜,幕后推手曝光被网友誉为&ldquo;纯欲天花板&rdquo;的女网红井川里予,近期因为一组哥特风照片登上热搜,引发了一场互联网世界关于
  • 东方甄选单飞:有些鸟注定是关不住的

    作者:彭宽鸿来源:华尔街科技眼&zwj;&zwj;&zwj;&zwj;&zwj;&zwj;&zwj;&zwj;&zwj;&zwj;东方甄选创始人俞敏洪带队的&ldquo;7天甘肃行&rdquo;直播活动已在近日顺利收官。成立后一
  • 小米汽车电池信息疑似曝光:容量101kWh,支持800V高压快充

    7月14日消息,今日一名博主在社交媒体发布了一张疑似小米汽车电池信息的照片,显示该电池包正是宁德时代麒麟电池,容量为101kWh,电压为726.7V,可以预测小
  • OPPO K11搭载长寿版100W超级闪充:26分钟充满100%

    据此前官方宣布,OPPO将于7月25日也就是今天下午14:30举办新品发布会,届时全新的OPPO K11将正式与大家见面,将主打旗舰影像,和同档位竞品相比,其最大的卖
Top