【Linux】POSIX 线程信号量与互斥锁▲
代码要求:高内聚,低耦合
高内聚:元素之间具有很强的关联性,模块的功能单一且集中
低耦合:代码之间的依赖关系尽可能简单,相互之间的影响和交互尽可能少
线程安全问题:多线程访问共享数据,且对共享数据的操作为非原子性操作(不可被中断的操作)
为解决这个问题:线程同步(线程信号量、线程互斥锁)
注意问题:线程同步带来的问题是效率的降低
重点注意:协议之间是数据传输,只能使用最基本的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;//获取当前线程idpthread_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;
}
相关文章:

【Linux】POSIX 线程信号量与互斥锁▲
代码要求:高内聚,低耦合 高内聚:元素之间具有很强的关联性,模块的功能单一且集中 低耦合:代码之间的依赖关系尽可能简单,相互之间的影响和交互尽可能少 线程安全问题:多线程访问共享数据&…...
轻松制作高质量视频,实时生成神器LTX-Video重磅登场!
探索LTX-Video:实时视频生成跨越新高度 在如今这个视觉内容主导的数字时代,视频生成成为推动创意表达的关键。而今天,我们将带您深入探索LTX-Video,一个强大的开源项目,致力于通过尖端技术将视频生成提升到一个全新的…...

USR-M100采集数据并提交MQTT服务器
本文为记录备忘,不做过多解释。 模块自身带有2路数字量输入,2路模拟量输入,2路485接口 数字量接报警输入,模拟量接压力传感器,液位传感器,485接口分别接流量计,温湿度传感器。 正确接线&…...

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

数据集-目标检测系列- 冥想 检测数据集 close_eye>> DataBall
数据集-目标检测系列- 冥想 检测数据集 close * 相关项目 1)数据集可视化项目:gitcode: https://gitcode.com/DataBall/DataBall-detections-100s/overview 2)数据集训练、推理相关项目:GitHub - XIAN-HHappy/ultralytics-yolo-…...
计算机网络:家庭路由器WiFi信号的发射和手机终端接收信号原理?
WiFi路由器与手机之间的信号传输涉及多个技术层面的协作,以下是其工作原理的详细步骤: 一、数据封装与协议处理 应用层数据生成 用户操作(如浏览网页、视频播放)产生数据包,经TCP/IP协议栈逐层封装,添加IP地址(网络层)和MAC地址(数据链路层)。协议封装 数据包被封装…...
用 NGINX 打造高性能 FastCGI 加速 `ngx_http_fastcgi_module`
一、安装与启用 # 在编译 NGINX 源码时加上: ./configure --with-http_fastcgi_module make && sudo make install# 或确保你使用的二进制已内置(大多数发行版都默认包含) nginx -V | grep fastcgi二、基础转发配置 http {server {…...
深度学习 ———— 迁移学习
迁移学习原理 什么是迁移学习? 迁移学习利用在大规模数据集(如ImageNet)上预训练的模型,改装小数据集(如CIFAR-10)。优势: 减少训练时间:预训练模型已学习通用特征(如边…...

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

以影像为笔,劳润智在世界舞台上书写艺术之路
在光影交织中,摄影师劳润智的镜头仿佛能穿透喧嚣,捕捉人类情感最细腻的脉动。从疫情下洛杉矶裁缝日常的温馨瞬间,到象征自由与解脱的飞鸟影像,再到探索时间与空间交错的抽象作品,每一幅作品都展现了他对艺术的深度追求与对生活的温柔洞察。 劳润智的作品为他赢得了多个国际奖项…...
vue3 computed方法传参数
我们对computed的基础用法不陌生,比如前端项目中经常会遇到数据处理的情况,我们就会选择computed方法来实现。但大家在碰到某些特殊场景,比如在template模板中for循环遍历时想给自己的计算属性传参,这个该怎么实现呢,很…...
【ES】Elasticsearch字段映射冲突问题分析与解决
在使用Elasticsearch作为搜索引擎时,经常会遇到一些映射(Mapping)相关的问题。本文将深入分析字段映射冲突问题,并通过原生的Elasticsearch API请求来复现和解决这个问题。 问题描述 在实际项目中,我们遇到以下错误: Transport…...
昇腾NPU容器内 apt 换源
环境 昊算NPU云910b 问题 缺少vim等,同时无法apt安装新的依赖 解决办法 使用vi修改/etc/apt/sources.list.d/debian.sources Types: deb URIs: http://deb.debian.org/debian Suites: bookworm bookworm-updates bookworm-backports Components: main contrib…...
MySQL 从入门到精通(五):索引深度解析 —— 性能优化的核心武器
目录 一、索引概述:数据库的 “目录” 1.1 什么是索引? 1.2 索引的性能验证:用事实说话 实验环境准备 无索引查询耗时 有索引查询耗时 索引的 “空间换时间” 特性 二、索引的创建:三种核心方式 2.1 方式 1:C…...
spark-Join Key 的基数/rand函数
在数据处理中,Join Key 的基数 是指 Join Key 的唯一值的数量(也称为 Distinct Key Count)。它表示某个字段(即 Join Key)在数据集中有多少个不同的值。 1. Join Key 基数的意义 高基数:Join Key 的唯一值…...

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

【桌面】【输入法】常见问题汇总
目录 一、麒麟桌面系统输入法概述 1、输入法介绍 2、输入法相关组件与服务 3、输入法调试相关命令 3.1、输入法诊断命令 3.2、输入法配置重新加载命令 3.3、启动fcitx输入法 3.4、查看输入法有哪些版本,并安装指定版本 3.5、重启输入法 3.6、查看fcitx进程…...
R语言学习--Day01--数据清洗初了解andR的经典筛选语法
当我们在拿到一份数据时,是否遇到过想要分析数据却无从下手?通过编程语言去利用它时发现有很多报错不是来源于代码而是因为数据里有很多脏数据;在这个时候,如果你会用R语言来对数据进行清洗,这会让你的效率提升很多。 …...

QT的初始代码解读及其布局和弹簧
this指的是真正的当前正在显示的窗口 main函数: Widget w是生成了一个主窗口,QT Designer是在这个主窗口里塞组件 w.show()用来展示这个主窗口 头文件: namespace Ui{class Widget;}中的class Widget和下面的class Widget不是一个东西 Ui…...

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

引用第三方自定义组件——微信小程序学习笔记
1. 使用 npm 安装第三方包 1.1 下载安装Node.js 工具 下载地址:Node.js — Download Node.js 1.2 安装 npm 包 在项目空白处右键弹出菜单,选择“在外部终端窗口打开”,打开命令行工具,输入以下指令: 1> 初始化:…...
Docker、Docker-compose、K8s、Docker swarm之间的区别
1.Docker docker是一个运行于主流linux/windows系统上的应用容器引擎,通过docker中的镜像(image)可以在docker中构建一个独立的容器(container)来运行镜像对应的服务; 例如可以通过mysql镜像构建一个运行mysql的容器,既可以直接进入该容器命…...

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

C#问题 加载格式不正确解决方法
出现上面问题 解决办法:C#问题 改成x86 不要选择anycpu...
VTK|结合qt创建通用按钮控制显隐(边框、坐标轴、点线面)
文章目录 增加边框BoundingBox添加addBoundingBox添加BoundingBox控制按钮点击按钮之后的槽函数 添加坐标轴增加点线面显隐控制按钮添加控制点线面显隐的按钮到三维显示界面控制面显示槽函数控制线显示槽函数控制点显示槽函数 增加边框BoundingBox 增加边框BoundingBox并通过按…...

CentOS 7.9 安装详解:手动分区完全指南
CentOS 7.9 安装详解:手动分区完全指南 为什么需要手动分区?CentOS 7.9 基本分区说明1. /boot/efi 分区2. /boot 分区3. swap 交换分区4. / (根) 分区 可选分区(进阶设置)5. /home 分区6. /var 分区7. /tmp 分区 分区方案建议标准…...
在过滤器中获取body中的json数据并且使得后续的controller层也能获取使用
前景提示: ①我需要在filter中获取到json数据->对key名首字母进行排序,然后拼接,进行验签 ②所以就需要在filer获取到json的数据,因为请求数据是一次性读取的流。如果过滤器中调用了request.json或request.get_json()ÿ…...

如何使用测试软件 Jmeter
第一步,点击 编辑 添加线程组 第二步,右键单击线程组,添加取样器 HTTP 请求 第三步,设置请求路径 第四步,添加 查看结果树 用于查看请求响应 最后点击绿色小三角启动即可...

2025盘古石初赛WP
来不及做,还有n道题待填坑 文章目录 手机取证 Mobile Forensics分析安卓手机检材,手机的IMSI是? [答案格式:660336842291717]养鱼诈骗投资1000,五天后收益是? [答案格式:123]分析苹果手机检材&a…...

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