Linux——多线程互斥

多线程互斥抢票问题及其解决方案

在多线程编程中,处理共享资源时可能会遇到互斥问题。让我们深入探讨一下这个问题,并通过一个具体的抢票场景来理解和解决它。

多线程抢票问题

假设我们有1000张火车票,四个线程同时在抢票。代码如下:

#include "Thread.hpp"int tickets = 1000; // 票数

void thread_run(void args) {string name = static_cast(args);while(true) {if(tickets > 0) {usleep(1234); // 1秒=1000毫秒=1000000微秒cout << name << "抢到票, 票数为: " << tickets-- << endl;} else {break;}}return nullptr;}

int main() {unique_ptr thread1(new Thread(thread_run, (void)"user1",1));unique_ptr thread2(new Thread(thread_run, (void)"user2",2));unique_ptr thread3(new Thread(thread_run, (void)"user3",3));unique_ptr thread4(new Thread(thread_run, (void)"user4",4));thread1->join();thread2->join();thread3->join();thread4->join();return 0;}

Linux——多线程互斥

问题分析

运行上述代码时,可能会看到票数出现0、-1、-2等负数的情况。这是由于多线程并发访问共享资源(票数)时,导致的数据竞争和不一致性。

立即进入“豆包AI人工智官网入口”;

立即学习“豆包AI人工智能在线问答入口”;

多线程交叉执行的本质是调度器频繁切换线程。线程切换可能在以下几种情况发生:

时间片用完更高优先级线程就绪线程进入等待状态

线程在从内核态切换到用户态时,会检测调度状态并可能进行线程切换。

原因解析

在读取和修改共享变量ticket的过程中,由于线程切换,可能会发生以下情况:

读取到寄存器:线程1读取票数到CPU寄存器中。判断和修改:如果在判断和修改之间发生线程切换,其他线程也可能读取到同样的票数,导致多个线程同时减少票数。

例如,假设四个线程同时读取到票数为1,然后线程1将票数减少到0并打印,之后其他线程也尝试减少票数,导致票数变成负数。

解决方案:使用互斥锁

为了解决上述问题,我们可以使用互斥锁来确保在修改共享资源时,只有单个线程能够访问。

使用全局锁

#include "Thread.hpp"int tickets = 1000; // 票数pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER; // 全局锁

void thread_run(void args) {string name = static_cast(args);while(true) {pthread_mutex_lock(&lock); // 加锁if(tickets > 0) {usleep(1234); // 1秒=1000毫秒=1000000微秒cout << name << "抢到票, 票数为: " << tickets-- << endl;} else {pthread_mutex_unlock(&lock);break;}pthread_mutex_unlock(&lock); // 解锁}return nullptr;}

int main() {unique_ptr thread1(new Thread(thread_run, (void)"user1",1));unique_ptr thread2(new Thread(thread_run, (void)"user2",2));unique_ptr thread3(new Thread(thread_run, (void)"user3",3));unique_ptr thread4(new Thread(thread_run, (void)"user4",4));thread1->join();thread2->join();thread3->join();thread4->join();return 0;}

Linux——多线程互斥

使用全局锁后,线程在进入临界区时会加锁,离开时解锁,确保了票数的正确性。然而,由于线程串行执行,速度可能会变慢。

使用局部锁和线程数据

为了进一步优化,我们可以使用局部锁和线程数据来管理每个线程的抢票过程:

#include "Thread.hpp"int tickets = 1000; // 票数

class ThreadData {public:ThreadData(const string& threadname, pthread_mutex_t mutex_p) : _threadname(threadname), _mutex_p(mutex_p) {}~ThreadData() {}public:string _threadname;pthread_mutex_t _mutex_p;};

void thread_run(void args) {ThreadData p = static_cast<ThreadData>(args);while(true) {pthread_mutex_lock(p->_mutex_p); // 加锁if(tickets > 0) {usleep(1234); // 1秒=1000毫秒=1000000微秒cout <_threadname << "抢到票, 票数为: " << tickets-- <_mutex_p);break;}pthread_mutex_unlock(p->_mutex_p); // 解锁usleep(1234); // 模拟抢完票形成一个订单}return nullptr;}

int main() {pthread_mutex_t lock;pthread_mutex_init(&lock, nullptr); // 初始化锁vector arr(4);for(int i = 0; i < 4; i++) {string thread_name = "user" + to_string(i + 1);ThreadData* data = new ThreadData(thread_name, &lock);pthread_create(&arr[i], nullptr, thread_run, data);}for(int i = 0; i < 4; i++) {pthread_join(arr[i], nullptr);}return 0;}

Linux——多线程互斥

锁的背景概念

临界资源:多线程共享的资源。临界区:访问临界资源的代码段。互斥:确保同一时间只有一个线程访问临界资源。原子性:操作要么完成,要么不完成,不会被打断。

锁的使用和原理

锁本身是共享资源,用于保护其他共享资源。pthread_mutex_lockpthread_mutex_unlock是原子操作,确保锁的安全性。

加锁和解锁的原理基于交换指令(如swapexchange),确保原子性。

锁的封装

为了兼容C++,我们可以封装锁的接口:

#pragma once

include

include

include

include

include

include

include

include

include

using namespace std;

class Mutex {public:Mutex(pthread_mutex_t lock_p = nullptr) : _lock_p(lock_p) {}void lock() { if(_lock_p) pthread_mutex_lock(_lock_p); }void unlock() { if(_lock_p) pthread_mutex_unlock(_lock_p); }~Mutex() {}private:pthread_mutex_t _lock_p;};

class LockGuard {public:LockGuard(pthread_mutex_t *mutex) : _mutex(mutex) { _mutex.lock(); }~LockGuard() { _mutex.unlock(); }private:Mutex _mutex;};

使用LockGuard可以实现RAII风格的自动加锁和解锁:

#include "Thread.hpp"

include "Mutex.hpp"

int tickets = 1000; // 票数

class ThreadData {public:ThreadData(const string& threadname, pthread_mutex_t mutex_p) : _threadname(threadname), _mutex_p(mutex_p) {}~ThreadData() {}public:string _threadname;pthread_mutex_t _mutex_p;};

void thread_run(void args) {ThreadData p = static_cast<ThreadData>(args);LockGuard lockGuard(p->_mutex_p); // 自动加锁解锁while(true) {if(tickets > 0) {usleep(1234); // 1秒=1000毫秒=1000000微秒cout <_threadname << "抢到票, 票数为: " << tickets-- << endl;} else {break;}}return nullptr;}

int main() {pthread_mutex_t lock;pthread_mutex_init(&lock, nullptr); // 初始化锁vector arr(4);for(int i = 0; i < 4; i++) {string thread_name = "user" + to_string(i + 1);ThreadData* data = new ThreadData(thread_name, &lock);pthread_create(&arr[i], nullptr, thread_run, data);}for(int i = 0; i < 4; i++) {pthread_join(arr[i], nullptr);}return 0;}

Linux——多线程互斥

线程安全与可重入

线程安全:多线程并发执行同一段代码时,结果一致。可重入:函数在被不同执行流调用时,结果不受影响。

死锁

死锁是指多个线程因互相等待对方释放资源而陷入永久等待的状态。死锁的四个必要条件是:

互斥:资源只能被一个线程占用。持有并等待:一个线程在持有资源的同时等待其他资源。不可剥夺:资源不能被强制从一个线程转移到另一个线程。循环等待:存在一个线程等待链,形成循环。

避免死锁的策略包括:

避免同时申请多个锁。使用锁的申请顺序。避免长时间持有锁

总之,多线程编程中的互斥问题可以通过使用锁来解决,但需要注意锁的使用可能会导致性能下降和死锁问题。

以上就是Linux——多线程互斥的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/70850.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月13日 07:52:29
下一篇 2025年11月13日 08:36:20

相关推荐

  • Go语言内存增长排查:time.Ticker的陷阱与正确使用姿势

    本文深入探讨了Go程序中因time.NewTicker在循环内重复创建而导致的内存持续增长问题。通过分析其内部机制,揭示了未停止旧Ticker实例如何引发资源泄露。教程提供了两种解决方案,并强调了将Ticker创建移至循环外进行复用的最佳实践,旨在帮助开发者避免此类常见的Go语言并发与资源管理陷阱。…

    2025年12月16日
    000
  • Nginx反向代理下Go应用重定向路径错误解决方案

    当Go应用在Nginx反向代理后进行重定向时,常出现跳转至服务器根目录而非应用自身根目录的问题。本文将深入分析此现象,并提供一种在Go应用层面配置基础路径并实现自定义重定向函数的方法,确保重定向行为符合预期,提升系统健壮性。 理解问题:Nginx反向代理与应用重定向 在微服务架构或多应用部署场景中,…

    2025年12月16日
    000
  • Go AST到源代码的转换:使用go/printer包生成Go源代码

    本文详细阐述如何利用Go语言标准库中的go/printer包,将抽象语法树(AST)转换回可执行的Go源代码。与go/parser用于解析源代码生成AST相辅相成,go/printer提供了一种将程序结构以AST形式表示后,再将其序列化为文本代码的有效方法。这对于实现代码生成、重构工具或静态分析后的…

    2025年12月16日
    000
  • Golang多层函数调用错误传递实践方法

    错误应在合适层级处理并清晰向上传递。Go使用error接口标准传递,每层检查错误并决定是否返回,如getUser中调用fetchFromDB,出错时用fmt.Errorf包装后向上返回。 在Go语言开发中,多层函数调用时的错误传递是一个常见且关键的问题。良好的错误处理机制不仅能提高程序的健壮性,还能…

    2025年12月16日
    000
  • Go语言AST到源代码的转换:使用go/printer

    本文详细介绍了在Go语言中如何将抽象语法树(AST)转换回可执行的源代码。通过利用标准库中的go/parser包解析源代码生成AST,并结合go/printer包的Fprint函数,开发者可以高效地实现AST到源代码的逆向生成,这对于代码分析、代码生成、重构或自动化工具开发至关重要。 在go语言的开…

    2025年12月16日
    000
  • Go 语言 len() 函数的性能探究:编译器优化揭秘

    Go 语言中对切片(slice)执行 len() 操作时,编译器会进行深度优化。它并非传统意义上的函数调用,而是被编译成直接访问内存中存储的长度值,效率极高。这意味着在循环条件中使用 len() 不会引入额外的性能开销,因为它只会在编译时或首次计算时被处理,后续迭代直接使用其结果,如同访问局部变量一…

    2025年12月16日
    000
  • Go语言中如何非递归地列出指定目录下的文件和文件夹

    本文详细介绍了如何在Go语言中使用os.ReadDir函数,以非递归方式高效地列出指定目录下的所有文件和子目录。文章提供了清晰的代码示例,演示了如何区分文件和目录,并讨论了错误处理及其他重要注意事项,帮助开发者准确地获取目录内容。 一、理解非递归目录列表的需求 在go语言中处理文件系统时,经常需要获…

    2025年12月16日
    000
  • Go 语言:非递归列出目录内容的实践指南

    本文将详细介绍如何在 Go 语言中非递归地列出指定目录下的文件和子目录。我们将重点使用 os 包中的 ReadDir 函数,并通过实例代码展示如何获取目录条目、区分文件与文件夹,并处理可能出现的错误,提供一种简洁高效的目录内容遍历方案。 在 go 语言中,处理文件系统操作是常见的需求。当我们需要获取…

    2025年12月16日
    000
  • Golang图像处理:深度解析PNG通道交换技术

    本文深入探讨了在Go语言中如何使用image和image/png包对PNG图像的颜色通道进行交换。我们将从image.Image接口的限制入手,逐步介绍两种核心的像素修改方法:通过自定义ImageSet接口进行通用像素操作,以及针对*image.RGBA类型的优化处理。文章将提供详细的代码示例,涵盖…

    2025年12月16日
    000
  • Golang环境搭建需要安装哪些工具

    答案:搭建Golang开发环境需安装Go SDK、配置环境变量、选择代码编辑器、安装Git及推荐工具。首先从官网下载Go SDK并设置GOROOT、PATH、GO111MODULE等环境变量;然后选用VS Code或GoLand等IDE提升效率;接着安装Git以支持模块依赖管理;最后可选gofmt、…

    2025年12月16日
    000
  • Golang Decorator装饰器模式功能扩展示例

    Go语言中通过接口组合和函数包装实现装饰器模式,用于日志、监控等功能扩展。首先定义UserService接口及其实现,接着创建loggingDecorator和metricsDecorator结构体,分别实现调用前后的日志记录与耗时统计。通过NewLoggingDecorator和NewMetric…

    2025年12月16日
    000
  • Golang反射获取结构体嵌套字段示例

    通过反射可逐层访问Go结构体嵌套字段,使用FieldByName获取字段值并检查IsValid避免panic,递归函数可处理任意深度嵌套,适用于导出字段的动态查询。 在Go语言中,反射(reflect)可以用来动态获取结构体字段信息,包括嵌套结构体的字段。通过reflect.Type和reflect…

    2025年12月16日
    000
  • Golang测试依赖隔离与mock技巧示例

    使用接口和mock技术可实现Go语言测试依赖隔离。通过定义UserRepository接口并创建MockUserRepo,结合testify/mock库动态模拟方法调用,能有效解耦外部依赖;利用httptest模拟HTTP服务响应,避免真实网络请求;借助sqlmock库mock数据库操作,提升测试效…

    2025年12月16日
    000
  • Golang适配器模式第三方接口兼容示例

    适配器模式通过定义统一SMSSender接口,为阿里云和腾讯云短信服务分别实现AliyunAdapter和TencentAdapter适配器,使不同SDK接口标准化,业务层可透明切换服务商,提升扩展性与维护性。 在使用 Golang 开发项目时,经常会对接第三方服务,比如支付、短信、物流等。不同第三…

    2025年12月16日
    000
  • Golang channel与context结合控制任务

    channel与context结合可实现协程的取消传播和超时控制,通过context.WithCancel或WithTimeout创建可取消上下文,在协程中监听ctx.Done()并用channel传递结果,主协程设置超时后能及时中断任务。 在 Go 语言中,channel 和 context 是实…

    2025年12月16日
    000
  • 如何优化Go与Android之间的数据传输:压缩策略与算法选择

    本文旨在探讨在Go服务器与Android客户端之间传输数据时,如何有效利用数据压缩技术。我们将分析不同数据类型(如媒体文件和文本)的压缩效益,并比较多种压缩算法(Deflate, Gzip, Bzip2, LZMA)在压缩比、计算成本和内存消耗方面的权衡,为开发者提供选择最佳压缩方案的指导。 在构建…

    2025年12月16日
    000
  • Go语言中fmt.Println()与println()的区别与最佳实践

    在Go语言中,fmt.Println()和内置函数println()都能实现控制台输出,但它们在设计目的、稳定性及使用场景上存在显著差异。fmt.Println()作为标准库fmt包的一部分,是Go应用程序进行通用输出和格式化打印的首选,具备高稳定性与丰富功能;而println()则是一个低层级的运…

    2025年12月16日
    000
  • Go语言结构体初始化:理解值类型与指针类型的选择

    在Go语言中,结构体是组织数据的重要方式。当我们初始化一个结构体时,常常会遇到两种看似相似但实则有本质区别的语法:StructName{} 和 &StructName{}。这两种初始化方式的核心差异在于它们所创建的变量类型不同,从而影响了程序的行为和内存管理。理解这一区别是Go语言编程的基础…

    2025年12月16日
    000
  • Go语言Cgo封装zlib库:解决deflateInit宏与链接问题

    本文探讨了如何在Go语言中使用Cgo封装C语言的zlib库,以提升压缩性能。重点解决了在调用deflateInit等宏时遇到的“未声明”错误,并指出了正确的库链接方法。通过创建一个C语言垫片函数来桥接Go和C宏,并配置Cgo链接参数,成功实现了zlib的集成与调用,为Go程序提供了高效的压缩能力。 …

    2025年12月16日
    000
  • Go语言中通过cgo封装zlib库:解决宏调用与链接问题

    本文详细介绍了如何在Go语言中利用cgo调用C语言的zlib库,并着重解决了在使用deflateInit等宏时遇到的常见问题。通过引入C语言辅助函数和正确的链接配置,我们展示了如何成功地在Go应用中集成zlib的压缩功能,为需要高性能或特定zlib特性的开发者提供了实用的解决方案。 1. 引言:Go…

    2025年12月16日
    000

发表回复

登录后才能评论
关注微信