iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > GO >解析golang中的并发安全和锁问题
  • 366
分享到

解析golang中的并发安全和锁问题

2024-04-02 19:04:59 366人浏览 独家记忆
摘要

1. 并发安全 package main import ( "fmt" "sync" ) var ( sum int wg sync.Wa

1. 并发安全


package main
 
import (
    "fmt"
    "sync"
)
 
var (
    sum int
    wg sync.WaitGroup
)
 
func test() {
    for i := 0; i < 5000000; i++ {
        sum += 1
    }
    wg.Done()
}
 
func main() {
    // 并发和安全
    wg.Add(2)
    Go test()
    go test()
 
    wg.Wait()
    fmt.Println(sum)
 
}

  上面的代码中我们开启了两个goroutine去累加变量x的值,这两个goroutine在访问和修改x变量的时候就会存在数据竞争,导致最后的结果与期待的不符。

2. 互斥锁


package main
 
import (
    "fmt"
    "sync"
)
 
var (
    sum int
    wg sync.WaitGroup
    mu sync.Mutex  // 定义一个互斥锁
)
 
func test() {
    for i := 0; i < 10000000; i++ {
        // 互斥锁它能够保证同时只能有一个goroutine去访问共享资源
        mu.Lock()
        sum += 1
        mu.Unlock()
    }
    wg.Done()
}
 
func main() {
    fmt.Println(mu)
    // 并发和安全锁
    wg.Add(2)
    go test()
    go test()
 
    wg.Wait()
    fmt.Println(sum)
 
}

  使用互斥锁能够保证同一时间有且只有一个goroutine进入临界区,其他的goroutine则在等待锁;当互斥锁释放后,等待的goroutine才可以获取锁进入临界区,多个goroutine同时等待一个锁时,唤醒的策略是随机的。

3. 读写互斥锁

互斥锁是完全互斥的,但是有很多实际的场景下是读多写少的,当我们并发的去读取一个资源不涉及资源修改的时候是没有必要加锁的,这种场景下使用读写锁是更好的一种选择。读写锁在Go语言中使用sync包中的RWMutex类型。

读写锁分为两种:读锁和写锁。当一个goroutine获取读锁之后,其他的goroutine如果是获取读锁会继续获得锁,如果是获取写锁就会等待;当一个goroutine获取写锁之后,其他的goroutine无论是获取读锁还是写锁都会等待。


package main
 
import (
    "fmt"
    "sync"
    "time"
)
 
var (
    x int
    wg sync.WaitGroup
    mu sync.Mutex  // 定义一个互斥锁
    rw sync.RWMutex  // 定义一个读写锁,注意:只有读多写少的时候,读写锁才能发挥其优势
)
 
func write() {
    rw.Lock()
    x += 1
    time.Sleep(10 * time.Millisecond)  // 假设写入时间耗费10毫秒
    rw.Unlock()
    wg.Done()
}
func read() {
    rw.RLock()
    time.Sleep(time.Millisecond)
    rw.RUnlock()
    wg.Done()
}
 
func main() {
    start := time.Now()
    for i := 0; i < 10; i++ {
        wg.Add(1)
        go write()
    }  // 写耗时:160毫秒左右
 
    for i := 0; i < 1000; i++ {
        wg.Add(1)
        go read()
    }  // 读耗时:15毫秒左右
 
    wg.Wait()
    end := time.Now()
    fmt.Println("执行时间:", end.Sub(start))
}

  需要注意的是读写锁非常适合读多写少的场景,如果读和写的操作差别不大,读写锁的优势就发挥不出来。

到此这篇关于golang中的并发安全和锁的文章就介绍到这了,更多相关golang并发和锁内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

您可能感兴趣的文档:

--结束END--

本文标题: 解析golang中的并发安全和锁问题

本文链接: https://www.lsjlt.com/news/156723.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 解析golang中的并发安全和锁问题
    1. 并发安全 package main import ( "fmt" "sync" ) var ( sum int wg sync.Wa...
    99+
    2024-04-02
  • 详解go语言中并发安全和锁问题
    首先可以先看看这篇文章,对锁有些了解 GO语言并发编程之互斥锁、读写锁详解 Mutex-互斥锁 Mutex 的实现主要借助了 CAS 指令 + 自旋 + 信号量 数据结构: ty...
    99+
    2024-04-02
  • golang函数的并发安全问题
    在 go 语言中,函数的并发安全性决定了是否可以在并发环境中安全调用该函数。并发安全问题可能出现于处理共享数据或修改内部状态时,如竞态条件和数据竞争。通过使用互斥锁或其他最佳实践(例如使...
    99+
    2024-04-28
    golang 并发安全问题 并发访问 标准库
  • 解析Golang中的锁竞争问题
    当我们打印错误的时候使用锁可能会带来意想不到的结果。 我们看下面的例子: package main import ( "fmt" "sync" ) type Coursewa...
    99+
    2024-04-02
  • Golang函数并发编程中的内存安全问题
    go 中的函数并发编程存在内存安全问题,解决方法包括:互斥锁:防止多个 goroutine 同时访问共享数据,通过锁定和解锁操作保护临界区。通道:用于 goroutine 之间安全传递值...
    99+
    2024-04-17
    并发编程 内存安全 golang 并发访问 同步机制
  • golang并发安全及读写互斥锁的示例分析
    目录并发安全和锁互斥锁读写互斥锁并发安全和锁 有时候在Go代码中可能会存在多个goroutine同时操作一个资源(临界区),这种情况会发生竞态问题(数据竞态)。类比现实生活中的例子有...
    99+
    2024-04-02
  • golang并发安全及锁怎么实现
    本文小编为大家详细介绍“golang并发安全及锁怎么实现”,内容详细,步骤清晰,细节处理妥当,希望这篇“golang并发安全及锁怎么实现”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。并发安全和锁有时候在Go代码中...
    99+
    2023-06-30
  • Golang的高并发场景中如何处理死锁和饥饿问题?
    死锁与饥饿在 go 并发编程中的成因和解决方法:死锁:由递归锁引起,避免方法是采用死锁避免算法。饥饿:由优先级反转引起,解决方法是使用优先级继承机制,让低优先级线程暂时获取高优先级线程的...
    99+
    2024-05-10
    死锁 饥饿 golang
  • C#中常见的并发集合和线程安全问题
    C#中常见的并发集合和线程安全问题在C#编程中,处理并发操作是非常常见的需求。当多个线程同时访问和修改同一数据时,就会出现线程安全问题。为了解决这个问题,C#提供了一些并发集合和线程安全的机制。本文将介绍C#中常见的并发集合以及如何处理线程...
    99+
    2023-10-22
    集合 并发 线程安全
  • Golang全局变量加锁的问题解决
    如果全局变量只读取 那自然是不需要加锁的 如果全局变量多进程读,多进程写,那自然是需要加读写锁的 但是如果全局变量只有一个进程写,其他进程读呢? 如果采用COW的方式,写进程只是通过...
    99+
    2024-04-02
  • 解析Go的Waitgroup和锁的问题
    学 Go 的时候知道 Go 语言支持并发,最简单的方法是通过 go 关键字开启 goroutine 即可。可在工作中,用的是 sync 包的 WaitGroup,然而这样还不够,当多...
    99+
    2024-04-02
  • golang函数的并发安全
    go 函数的并发安全是指函数在并发调用时仍能正确运作,避免因多个 goroutine 同时访问数据而导致损坏。实现并发安全的函数可以使用锁、通道或原子变量等方法。锁允许 goroutin...
    99+
    2024-04-20
    golang 并发安全 同步机制
  • Golang开发注意事项:如何处理并发安全性问题
    在当今互联网时代,由于系统需求复杂度的增加,对高并发性能和安全的要求也变得越发迫切。Golang作为一种并发编程语言,以其简洁高效的特性而备受青睐。然而,开发人员在使用Golang进行并发编程时,必须时刻关注并处理并发安全性问题。在本文中,...
    99+
    2023-11-22
    Golang 并发 安全性
  • 解决Golang并发工具Singleflight的问题
    目录前言定义用途简单Demo源码分析结构对外暴露的方法重点方法分析Do流程图ForgetdoCall实际使用弊端与解决方案参考文章前言 前段时间在一个项目里使用到了分布式锁进行共享资...
    99+
    2024-04-02
  • 解决Golang Map 并发访问问题的方法
    一分耕耘,一分收获!既然打开了这篇文章《解决Golang Map 并发访问问题的方法》,就坚持看下去吧!文中内容包含等等知识点...希望你能在阅读本文后,能真真实实学到知识或者帮你解决心中的疑惑,也...
    99+
    2024-04-04
  • 如何处理PHP开发中的并发操作和线程安全问题
    在PHP开发中,处理并发操作和线程安全问题是非常重要的。特别是在高并发的场景下,如何保证数据的一致性和正确性是一个必须要解决的难题。本文将介绍一些常见的处理并发操作和线程安全问题的方法,并附上具体的代码示例。一、乐观锁和悲观锁乐观锁是一种乐...
    99+
    2023-10-21
    并发操作 - 处理PHP并发 线程安全 - PHP线程安全
  • golang中并发和并行的示例分析
    这篇文章主要介绍了golang中并发和并行的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。什么是golanggolang 是Google开发的一种静态强类型、编译型、...
    99+
    2023-06-15
  • golang高并发问题怎么解决
    在Go语言中,高并发问题可以通过以下几种方式来解决: 使用goroutine:Goroutine是Go语言的轻量级线程,可以并发...
    99+
    2024-02-29
    golang
  • Java并发问题之乐观锁与悲观锁的示例分析
    这篇文章将为大家详细讲解有关Java并发问题之乐观锁与悲观锁的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。首先介绍一些乐观锁与悲观锁:悲观锁:总是假设最坏的情况,每次去拿数据的时候都认为别人会修...
    99+
    2023-05-30
    java
  • Golang并发操作中常见的读写锁详析
    互斥锁简单粗暴,谁拿到谁操作。今天给大家介绍一下读写锁,读写锁比互斥锁略微复杂一些,不过我相信我们今天能够把他拿下! golang读写锁,其特征在于 读锁:可以同时进行多个...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作