【Linux】POSIX 线程信号量与互斥锁▲

news2025/5/11 16:41:28

代码要求:高内聚,低耦合

高内聚:元素之间具有很强的关联性,模块的功能单一且集中

低耦合:代码之间的依赖关系尽可能简单,相互之间的影响和交互尽可能少

线程安全问题:多线程访问共享数据,且对共享数据的操作为非原子性操作(不可被中断的操作)

为解决这个问题:线程同步(线程信号量、线程互斥锁)

注意问题:线程同步带来的问题是效率的降低

重点注意:协议之间是数据传输,只能使用最基本的char [ ],不能使用指针(char  *)、容器(list、string等),因为是两个客户端之间的连接传输,相当于两台电脑,并不能知道你发送的地址是什么  !!!

这些是 POSIX 线程库中多线程技术相关的函数,若执行成功,通常返回值为 0;若执行失败,会返回一个非零的错误码

一、线程信号量

1.信号量初始化

int sem_init(sem_t *sem,int pshared,unsigned value);

参数:

sem: 要进行初始化的信号量对象
pshared :控制着信号量的类型,如果值为 0,表示它是当前进程的局部信号量
value:赋给信号量对象的一个整数类型的初始值

2.解锁

// 给信号量的值加上一个“1”
int  sem_post (sem_t *sem);

3.加锁

//信号量的值减去一个“1”

int  sem_wait (sem_t *sem);

4.销毁

int  sem_destroy (sem_t *sem);

二、线程互斥量

1.初始化

int pthread_mutex_init (pthread_mutex_t* mutex, const pthread_mutexattr_t* mutexattr);

2.加锁

int pthread_mutex_lock (pthread_mutex_t* mutex);

3.解锁

int pthread_mutex_unlock(pthread_mutex_t* mutex);

4.销毁

int pthread_mutex_destroy(pthread_mutex_t* mutex);

三、两者之间的区别:

1. 资源访问数量限制
  • 线程信号量:可以允许多个线程同时访问共享资源,具体数量由信号量的初始值决定。例如,若信号量初始值为 3,那么最多允许 3 个线程同时访问共享资源。
  • 线程互斥锁:同一时间只允许一个线程访问共享资源,起到独占访问的作用。
2. 用途场景
  • 线程信号量:适用于多个资源实例的情况,如多个数据库连接、多个文件句柄等。它可以控制并发访问的线程数量,避免资源耗尽。
  • 线程互斥锁:主要用于保护临界区,防止多个线程同时访问共享资源而导致的数据不一致问题,如对共享变量的读写操作。

四、条件变量

条件变量用来自动阻塞一个线程,直到某特殊情况发生为止。通常条件下 变量和互斥锁同时使用
条件变量是用来等待事件

这些 POSIX 线程库中与条件变量相关的函数,若执行成功,通常返回值为 0;若执行失败,会返回一个非零的错误码

(1)初始化

//初始化条件变量

int  pthread_cond_init(pthread_cond_t  * cond,pthread_condattr_t  * cond_attr)

(2)等待

 //自动释放mutex锁,等待条件满足

int  pthread_cond_wait(pthread_cond_t  * cond,pthread_mutex_t  * mutex);
//自动释放 mutex 锁,等待条件满足,如果在 abstime 时间内还没有满足,则返回错误
int  pthread_cond_timewait(pthread_cond_t  * cond,pthread_mutex  * mutex,const  timespec  * abstime);

(3)销毁

//销毁条件变量
int  pthread_cond_destroy(pthread_cond_t  * cond);  

(4)唤醒

//让等待条件满足的线程中某一个被唤醒

int  pthread_cond_signal(pthread_cond_t  * cond);

//让等待条件满足的线程中某一个被唤醒

int  pthread_cond_broadcast(pthread_cond_t  * cond);

五、线程池

一个系统一般达到300线程是做合适的,超过300之后,性能反倒是降低,因此可以使用线程池

  • 减少线程创建和销毁的开销:线程的创建和销毁是比较耗时的操作,若频繁进行,会浪费大量的系统资源和时间。线程池在初始化时创建一定数量的线程,这些线程可以被重复使用来执行不同的任务,避免了反复创建和销毁线程的开销,从而提高了系统的响应速度和性能。

  • 提高资源利用率:线程池可以根据系统的负载情况动态调整线程的数量。当有大量任务提交时,线程池会增加线程数量来处理任务;当任务较少时,线程池会减少线程数量,避免资源的浪费。这样可以使系统资源得到更充分的利用,提高系统的整体性能。

  • 提高响应速度:当有任务到达时,线程池中有空闲线程可以立即执行任务,而不需要等待线程的创建

(1)首先线程池需要一个任务队列,用来存放客户端发来的请求,

(2)两个链表,一个空闲链表(将初始化的线程全部放在里面)

(3)一个忙碌链表,存放正在干活的线程

(4)通过条件变量来控制进程等待还是唤醒

(5)通过互斥锁来避免高并发带来的同时修改同一数据问题

#include "ThreadPool2.h"

ThreadPool2::ThreadPool2(int num)
{
	this->min_num = num;
	pthread_mutex_init(&this->mutex, NULL);
	pthread_cond_init(&this->conn, NULL);
	for (int i = 0; i < this->min_num; i++)
	{
		pthread_t id = 0;
		pthread_create(&id, NULL, thread_handle, this);
		this->idle_list.push_back(id);
	}
}

ThreadPool2::~ThreadPool2()
{
}

void ThreadPool2::lock()
{
	pthread_mutex_lock(&this->mutex);
}

void ThreadPool2::unlock()
{
	pthread_mutex_unlock(&this->mutex);
}

void ThreadPool2::wait()
{
	pthread_cond_wait(&this->conn, &this->mutex);
}

//唤醒线程池
void ThreadPool2::wakeup()
{
	pthread_cond_signal(&this->conn);

}

//添加任务到队列
void ThreadPool2::add_task(BaseTask* task)
{
	this->task_queue.push(task);
	this->wakeup();
}

//从队列取出一个任务
BaseTask* ThreadPool2::remove_task()
{

	BaseTask* task = this->task_queue.front();
	this->task_queue.pop();

	return task;
}

//从空闲队列到忙碌队列
void ThreadPool2::idle_to_busy(pthread_t id)
{

	list<pthread_t>::iterator it;
	it = find(this->idle_list.begin(), this->idle_list.end(), id);
	if (it != this->idle_list.end())
	{
		this->idle_list.erase(it);//从空闲队列移除
		this->busy_list.push_back(id);//添加到忙碌队列中
	}

}

void ThreadPool2::busy_to_idle(pthread_t id)
{
	list<pthread_t>::iterator it;
	it = find(this->busy_list.begin(), this->busy_list.end(), id);
	if (it != this->busy_list.end())
	{
		this->busy_list.erase(it);
		this->idle_list.push_back(id);
	}
}

void* ThreadPool2::thread_handle(void* p)
{
	ThreadPool2* p_this = (ThreadPool2*)p;
	//获取当前线程id
	pthread_t id = pthread_self();

	//将当前线程与主线程脱离
	pthread_detach(id);

	while (1)
	{
		
		p_this->lock();
		if (p_this->task_queue.empty())
		{
			p_this->wait();
		}

		//将当前线程从空闲列表移动到忙碌列表
		p_this->idle_to_busy(id);
		//从任务队列中获取一个任务
		BaseTask* task = p_this->remove_task();
		p_this->unlock();
		//执行任务
		task->working();
		cout << "任务执行完毕" << endl;
		p_this->busy_to_idle(id);
	}

	return nullptr;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2373260.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

USR-M100采集数据并提交MQTT服务器

本文为记录备忘&#xff0c;不做过多解释。 模块自身带有2路数字量输入&#xff0c;2路模拟量输入&#xff0c;2路485接口 数字量接报警输入&#xff0c;模拟量接压力传感器&#xff0c;液位传感器&#xff0c;485接口分别接流量计&#xff0c;温湿度传感器。 正确接线&…

内网穿透系列三:开源本地服务公网映射工具 tunnelmole

以下是对 tunnelmole 简要介绍&#xff1a; tunnelmole 是一款开源的内网穿透工具&#xff0c;一行命令就能把本地http服务映射成公网可访问的链接提供公共免费的网络服务&#xff0c;直接下载运行命令即可使用&#xff0c;也支持自行配置搭建私有客户端、服务端参考开源地址&…

数据集-目标检测系列- 冥想 检测数据集 close_eye>> DataBall

数据集-目标检测系列- 冥想 检测数据集 close * 相关项目 1&#xff09;数据集可视化项目&#xff1a;gitcode: https://gitcode.com/DataBall/DataBall-detections-100s/overview 2&#xff09;数据集训练、推理相关项目&#xff1a;GitHub - XIAN-HHappy/ultralytics-yolo-…

论文精读:YOLOE: Real-Time Seeing Anything

文章目录 前言1、背景2、方法2.1.重参Region-Text对齐模块2.2.VisualPrompt模块2.3.PromptFree 2.4.损失函数3、实验3.1.训练集3.2.实验结果 总结 前言 本文介绍一篇来自清华的开放词汇检测论文&#xff1a;YOLOE&#xff1b;源码链接。 1、背景 本文在yolo-world基础上&#x…

以影像为笔,劳润智在世界舞台上书写艺术之路

在光影交织中,摄影师劳润智的镜头仿佛能穿透喧嚣,捕捉人类情感最细腻的脉动。从疫情下洛杉矶裁缝日常的温馨瞬间,到象征自由与解脱的飞鸟影像,再到探索时间与空间交错的抽象作品,每一幅作品都展现了他对艺术的深度追求与对生活的温柔洞察。 劳润智的作品为他赢得了多个国际奖项…

LLMs之ChatGPT:《Connecting GitHub to ChatGPT deep research》翻译与解读

LLMs之ChatGPT&#xff1a;《Connecting GitHub to ChatGPT deep research》翻译与解读 导读&#xff1a;这篇OpenAI帮助文档全面介绍了将GitHub连接到ChatGPT进行深度代码研究的方法、优势和注意事项。通过连接GitHub&#xff0c;用户可以充分利用ChatGPT强大的代码理解和生成…

【桌面】【输入法】常见问题汇总

目录 一、麒麟桌面系统输入法概述 1、输入法介绍 2、输入法相关组件与服务 3、输入法调试相关命令 3.1、输入法诊断命令 3.2、输入法配置重新加载命令 3.3、启动fcitx输入法 3.4、查看输入法有哪些版本&#xff0c;并安装指定版本 3.5、重启输入法 3.6、查看fcitx进程…

QT的初始代码解读及其布局和弹簧

this指的是真正的当前正在显示的窗口 main函数&#xff1a; Widget w是生成了一个主窗口&#xff0c;QT Designer是在这个主窗口里塞组件 w.show()用来展示这个主窗口 头文件&#xff1a; namespace Ui{class Widget;}中的class Widget和下面的class Widget不是一个东西 Ui…

Profinet转CanOpen网关,打破协议壁垒的关键技术

在石油化工行业的生产现场&#xff0c;各类自动化设备如同精密运转的神经系统&#xff0c;而通信协议则是传递信号的"语言"。当不同厂商的设备采用Canopen与Profinet这两种主流工业协议时&#xff0c;就像两个使用不同方言的专家需要实时协作&#xff0c;此时开疆智能…

引用第三方自定义组件——微信小程序学习笔记

1. 使用 npm 安装第三方包 1.1 下载安装Node.js 工具 下载地址&#xff1a;Node.js — Download Node.js 1.2 安装 npm 包 在项目空白处右键弹出菜单&#xff0c;选择“在外部终端窗口打开”&#xff0c;打开命令行工具&#xff0c;输入以下指令&#xff1a; 1> 初始化:…

SpringAI实现AI应用-使用redis持久化聊天记忆

SpringAI实战链接 1.SpringAl实现AI应用-快速搭建-CSDN博客 2.SpringAI实现AI应用-搭建知识库-CSDN博客 3.SpringAI实现AI应用-内置顾问-CSDN博客 4.SpringAI实现AI应用-使用redis持久化聊天记忆-CSDN博客 5.SpringAI实现AI应用-自定义顾问&#xff08;Advisor&#xff09…

C#问题 加载格式不正确解决方法

出现上面问题 解决办法&#xff1a;C#问题 改成x86 不要选择anycpu

CentOS 7.9 安装详解:手动分区完全指南

CentOS 7.9 安装详解&#xff1a;手动分区完全指南 为什么需要手动分区&#xff1f;CentOS 7.9 基本分区说明1. /boot/efi 分区2. /boot 分区3. swap 交换分区4. / (根) 分区 可选分区&#xff08;进阶设置&#xff09;5. /home 分区6. /var 分区7. /tmp 分区 分区方案建议标准…

如何使用测试软件 Jmeter

第一步&#xff0c;点击 编辑 添加线程组 第二步&#xff0c;右键单击线程组&#xff0c;添加取样器 HTTP 请求 第三步&#xff0c;设置请求路径 第四步&#xff0c;添加 查看结果树 用于查看请求响应 最后点击绿色小三角启动即可

2025盘古石初赛WP

来不及做&#xff0c;还有n道题待填坑 文章目录 手机取证 Mobile Forensics分析安卓手机检材&#xff0c;手机的IMSI是&#xff1f; [答案格式&#xff1a;660336842291717]养鱼诈骗投资1000&#xff0c;五天后收益是&#xff1f; [答案格式&#xff1a;123]分析苹果手机检材&a…

系统分析与设计期末复习

第一章 系统的五个特性 整体性、目的性、相关性、环境适应性、层次性 软件系统的四个特性 复杂性、一致性、可变性、不可见性 第二章 系统规划 系统开发生命周期 系统规划->系统分析->系统设计->系统实施->系统运行维护->系统规划 诺兰阶段模型 阶段&a…

IBM BAW(原BPM升级版)使用教程第八讲

续前篇&#xff01; 一、流程开发功能模块使用逻辑和顺序 前面我们已经对 流程、用户界面、公开的自动化服务、服务、事件、团队、数据、性能、文件各个模块进行了详细讲解&#xff0c;现在统一进行全面统一讲解。 在 IBM Business Automation Workflow (BAW) 中&#xff0c;…

从电动化到智能化,法雷奥“猛攻”中国汽车市场

当前&#xff0c;全球汽车产业正在经历前所未有的变革&#xff0c;外资Tier1巨头开始向中国智能电动汽车市场发起新一轮“猛攻”。 在4月23日-5月2日上海国际车展期间&#xff0c;博世、采埃孚、大陆集团、法雷奥等全球百强零部件厂商纷纷发布战略新品与转型计划。在这其中&am…

鸿蒙开发——3.ArkTS声明式开发:构建第一个ArkTS应用

鸿蒙开发——3.ArkTS声明式开发:构建第一个ArkTS应用 一、创建ArkTS工程二、ArkTS工程目录结构&#xff08;Stage模型&#xff09;三、构建第一个页面四、构建第二个页面五、实现页面之间的跳转六、模拟器运行 一、创建ArkTS工程 1、若首次打开DevEco Studio&#xff0c;请点击…

word换行符和段落标记

换行符&#xff1a;只换行不分段 作用&#xff1a;我们需要对它进行分段&#xff0c;但它是一个信息群组&#xff0c;我希望它们有同样的段落格式&#xff01; 快捷键&#xff1a;shiftenter 段落标记&#xff1a;分段 快捷键&#xff1a;enter 修改字体格式或段落格式 …