Linux 信号量
Linux 信号量
- 一、信号量基础概念
- 1.1 同步机制的核心需求
- 1.2 信号量的核心原理
- 1.3 信号量类型对比
- 二、实战代码解析
- 2.1 共享内存与信号量结合示例
- 2.2 信号量类实现要点
- 三、关键实现细节分析
- 3.1 初始化三步骤
- 3.2 SEM_UNDO机制
- 3.3 原子操作保证
- 四、进阶应用场景
- 4.1 生产者-消费者模型
- 4.2 读写锁实现
- 五、最佳实践建议
- 六、常见问题排查
- 七、现代替代方案
一、信号量基础概念
1.1 同步机制的核心需求
在多进程/多线程编程中,当多个执行单元需要访问共享资源时,必须引入同步机制来保证数据一致性。信号量(Semaphore)正是解决这一问题的经典方案。
1.2 信号量的核心原理
信号量本质上是一个计数器,通过两个原子操作实现进程同步:
- P操作(wait):申请资源,计数器减1
- V操作(post):释放资源,计数器加1
1.3 信号量类型对比
| 类型 | System V信号量 | POSIX信号量 |
|---|---|---|
| 初始化方式 | 需要显式初始化 | 可静态初始化 |
| 作用域 | 系统级 | 进程级 |
| 性能 | 较高开销 | 较低开销 |
| 功能复杂度 | 支持信号量集合 | 仅支持单个信号量 |
二、实战代码解析
2.1 共享内存与信号量结合示例
// 演示使用信号量给共享内存加锁
#include "_public.h"struct stgirl {int no;char name[51];
};int main(int argc, char* argv[]) {if (argc != 3) {cout << "Usage: ./test no name\n";return -1;}// 创建/获取共享内存int shmid = shmget(0x5005, sizeof(stgirl), 0640|IPC_CREAT);stgirl* ptr = (stgirl*)shmat(shmid, 0, 0);// 初始化信号量csemp mutex;mutex.init(0x5005);// 临界区保护cout << "申请加锁...\n";mutex.wait();// 操作共享数据cout << "原值: no=" << ptr->no << ", name=" << ptr->name << endl;ptr->no = atoi(argv[1]);strcpy(ptr->name, argv[2]);sleep(10); // 模拟耗时操作mutex.post();shmdt(ptr);return 0;
}
2.2 信号量类实现要点
class csemp {
public:bool init(key_t key, unsigned short value=1, short sem_flg=SEM_UNDO);bool wait(short sem_op=-1);bool post(short sem_op=1);// ...其他成员函数
private:int m_semid;short m_sem_flg;
};// 初始化流程图
graph TDA[开始] --> B{信号量是否存在?}B -- 存在 --> C[直接获取]B -- 不存在 --> D[创建新信号量]D --> E[设置初始值]E --> F[初始化完成]
三、关键实现细节分析
3.1 初始化三步骤
- 尝试获取现有信号量
- 创建新信号量(IPC_EXCL保证原子性)
- 设置初始值(仅创建者需要)
3.2 SEM_UNDO机制
- 作用:防止进程异常终止导致的死锁
- 实现方式:内核维护调整记录
- 适用场景:建议用于互斥锁场景
3.3 原子操作保证
struct sembuf sem_b;
sem_b.sem_num = 0;
sem_b.sem_op = -1; // P操作
sem_b.sem_flg = SEM_UNDO;
semop(m_semid, &sem_b, 1);
四、进阶应用场景
4.1 生产者-消费者模型
// 初始化两个信号量
csemp empty(10); // 缓冲区空位
csemp full(0); // 已填充数量// 生产者
empty.wait();
// 生产数据...
full.post();// 消费者
full.wait();
// 消费数据...
empty.post();
4.2 读写锁实现
csemp mutex(1); // 互斥锁
csemp writeLock(1);// 写锁
int readers = 0;// 读锁定
mutex.wait();
readers++;
if (readers == 1) writeLock.wait();
mutex.post();// 读解锁
mutex.wait();
readers--;
if (readers == 0) writeLock.post();
mutex.post();// 写锁定
writeLock.wait();// 写解锁
writeLock.post();
五、最佳实践建议
-
命名规范
- 使用ftok生成唯一key
- 示例:
key_t key = ftok("/tmp", 'A');
-
错误处理
if (semop(...) == -1) {if (errno == EINTR) {// 处理信号中断}// 其他错误处理 } -
性能优化
- 优先考虑POSIX信号量
- 避免过度使用信号量集合
- 设置合理的超时机制
-
调试技巧
- 使用
ipcs -s查看信号量状态 - 通过
semctl获取当前值
- 使用
六、常见问题排查
-
ENOSPC错误
- 原因:系统信号量数量达到上限
- 解决:
sysctl -w kernel.sem="250 32000 100 128"
-
EIDRM错误
- 现象:信号量被意外删除
- 预防:增加引用计数机制
-
死锁检测
- 使用
pstack分析进程堆栈 - 借助valgrind工具链检测
- 使用
七、现代替代方案
-
原子变量
std::atomic<int> counter(0); counter.fetch_add(1, std::memory_order_relaxed); -
文件锁
int fd = open("lockfile", O_CREAT|O_RDWR, 0644); flock(fd, LOCK_EX); -
RCU机制
- 适用读多写少场景
- 无锁读取设计
信号量作为经典的进程同步工具,在系统级编程中仍具有重要地位。理解其底层机制并结合现代编程范式,能够帮助开发者构建更健壮的并发系统。在实际应用中,需要根据具体场景选择最合适的同步策略,平衡性能与安全性。
相关文章:
Linux 信号量
Linux 信号量 一、信号量基础概念1.1 同步机制的核心需求1.2 信号量的核心原理1.3 信号量类型对比 二、实战代码解析2.1 共享内存与信号量结合示例2.2 信号量类实现要点 三、关键实现细节分析3.1 初始化三步骤3.2 SEM_UNDO机制3.3 原子操作保证 四、进阶应用场景4.1 生产者-消费…...
Qt开发①Qt的概念+发展+优点+应用+使用
目录 1. Qt的概念和发展 1.1 Qt的概念 1.2 Qt 的发展史: 1.3 Qt 的版本 2. Qt 的优点和应用 2.1 Qt 的优点: 2.2 Qt 的应用场景 2.3 Qt 的应用案例 3. 搭建 Qt 开发环境 3.1 Qt 的开发工具 3.2 Qt SDK 的下载和安装 3.3 Qt 环境变量配置和使…...
向量库(Vector Database)
向量库 1. 向量库发展史 早期阶段(2000s) 基于关系型数据库的扩展(如 PostgreSQL 的向量插件)。简单相似度计算(如欧氏距离、余弦相似度)。 专用向量库的兴起(2010s) FAISS…...
torchsparse安装过程的问题
1、项目要求torchsparse githttps://github.com/mit-han-lab/torchsparse.gitv1.4.0 2、torch1.8.1cu111 nvcc--version:11.1 这个版本的cuda匹配的gcc、g经常是7.5。设置为7.5. (这个gcc、g版本修改不一定,可以先进行后面的,…...
【核心算法篇七】《DeepSeek异常检测:孤立森林与AutoEncoder对比》
大家好,今天我们来深入探讨一下《DeepSeek异常检测:孤立森林与AutoEncoder对比》这篇技术博客。我们将从核心内容、原理、应用场景等多个方面进行详细解析,力求让大家对这两种异常检测方法有一个全面而深入的理解。 一、引言 在数据科学和机器学习领域,异常检测(Anomaly…...
Win10环境使用零讯ZeroNews内网穿透实现Deepseek对外服务
Win10环境使用零讯ZeroNews内网穿透实现Deepseek对外服务 前言 之前笔者已经在Win10环境搭建好了Ollama、DeepSeek、Open WebUI、Dify等组件,成功实现了私有化部署及内网访问: https://lizhiyong.blog.csdn.net/article/details/145505686 https://l…...
CUDA 安装 一直卡在Installing Nsight Visual Studio Edition
最近在安装CUDA的时候,CUDA 安装 一直卡在Installing Nsight Visual Studio Edition,莫名的一直卡在安装进行中这儿,过很久都没进度,如图 后面重新下载了12.6的进行安装也是如此 无论是local还是network,都是这样。度…...
Softing线上研讨会 | 自研还是购买——用于自动化产品的工业以太网
| 线上研讨会时间:2025年1月27日 16:00~16:30 / 23:00~23:30 基于以太网的通信在工业自动化网络中的重要性日益增加。设备制造商正面临着一大挑战——如何快速、有效且经济地将工业以太网协议集成到其产品中。其中的关键问题包括:是否只需集成单一的工…...
STM32 定时器产生定周期方法
目录 背景 程序 第一步、使能PCLK1外设时钟编辑 第二步、时基单元配置 第三步、配置NVIC(设置定时中断优先级)编辑 第四步、使能溢出中断 第五步、使能定时器 第六步、填写中断处理函数(ISR) 背景 在单片机开发当中&…...
解锁机器学习核心算法 | 支持向量机:机器学习中的分类利刃
一、引言 在机器学习的庞大算法体系中,有十种算法被广泛认为是最具代表性和实用性的,它们犹如机器学习领域的 “十大神器”,各自发挥着独特的作用。这十大算法包括线性回归、逻辑回归、决策树、随机森林、K - 近邻算法、K - 平均算法、支持向…...
青少年编程与数学 02-009 Django 5 Web 编程 21课题、部署
青少年编程与数学 02-009 Django 5 Web 编程 21课题、部署 一、软件开发部署部署的主要内容部署的步骤部署的方式部署的环境 二、Django项目部署1. 准备工作2. 代码部署3. 配置Django项目4. Web服务器和应用服务器配置5. 安全和性能优化6. 监控和日志管理7. 测试和上线 三、在U…...
ARM系统源码编译OpenCV 4.10.0(包含opencv_contrib)
因项目部署在ARM系统上,需要编译一个arm版本的opencv-4.10.0(带opencv_contrib)版本。 若需要Linux系统下源码安装OpenCV,可参考:https://blog.csdn.net/qq_45445740/article/details/142770493?spm1001.2014.3001.55…...
cmake:定位Qt的ui文件
如题。在工程中,将h,cpp,ui文件放置到不同文件夹下,会存在cmake找不到ui文件,导致编译报错情况。 cmake通过指定文件路径,确保工程找到ui文件。 标识1:ui文件保存路径。 标识2:添加…...
(leetcode 1749 前缀和)1749. 任意子数组和的绝对值的最大值
核心题意 任意子数组和 的绝对值的最大值实际上是前缀和之间的差的最大值 建立前缀和数组 如果我们只考虑前缀和的最大值和最小值之差,那么就能够获得一个最大的子数组和的绝对值。因为任意一个子数组的和 prefix[j1] - prefix[i],它的绝对值是最大当…...
下载安装运行测试开源vision-language-action(VLA)模型OpenVLA
1. 安装 项目官网OpenVLA 首先按照官网提示的以下代码,执行创建环境->安装最小依赖->git克隆项目等 # Create and activate conda environment conda create -n openvla python3.10 -y conda activate openvla# Install PyTorch. Below is a sample comma…...
【网络安全 | 漏洞挖掘】我如何通过Cookie Manipulation发现主域上的关键PII?
未经许可,不得转载。 文章目录 正文正文 在分析 Example.com 的认证机制时,我注意到一个特定的 cookie,USER_ID,包含了一个具有预测性的会话标识符,其格式为: USER_ID="VYCVCDs-TZBI:XXXX-random-data"其中,XXXX 是由四个大写字母组成的部分,我使用 Burp S…...
【操作系统】操作系统概述
操作系统概述 1.1 操作系统的概念1.1.1 操作系统定义——什么是OS?1.1.2 操作系统作用——OS有什么用?1.1.3 操作系统地位——计算机系统中,OS处于什么地位?1.1.4 为什么学操作系统? 1.2 操作系统的历史1.2.1 操作系统…...
SQL Server 运算符优先级
在 SQL Server 中,运算符的优先级决定了在没有使用括号明确指定计算顺序时,运算符的执行顺序。 运算符优先级列表 括号 () 一元运算符 (正号)-(负号)~(按位取反) 乘法、除法和取模…...
Python的顺序结构和循环结构
文章目录 一、条件语句(1)条件语句的定义(2)条件语句的语法(a)单分支 if(b)双分支 if-else(c)多分支 if-elif-elif-...-else (3)注意事…...
深入浅出TypedArray:网络数据处理、WebGPU与加密实战
JavaScript的TypedArray是现代Web开发中处理二进制数据的利器。本文将结合网络数据传输、WebGPU编程和简单加密算法三个实战场景,带你领略TypedArray的强大能力。 一、TypedArray基础认知 TypedArray家族包括Int8Array、Uint16Array、Float32Array等11种视图类型&a…...
MySQL 隔离级别:脏读、幻读及不可重复读的原理与示例
一、MySQL 隔离级别 MySQL 提供了四种隔离级别,用于控制事务之间的并发访问以及数据的可见性,不同隔离级别对脏读、幻读、不可重复读这几种并发数据问题有着不同的处理方式,具体如下: 隔离级别脏读不可重复读幻读性能特点及锁机制读未提交(READ UNCOMMITTED)允许出现允许…...
JavaScript 中的 ES|QL:利用 Apache Arrow 工具
作者:来自 Elastic Jeffrey Rengifo 学习如何将 ES|QL 与 JavaScript 的 Apache Arrow 客户端工具一起使用。 想获得 Elastic 认证吗?了解下一期 Elasticsearch Engineer 培训的时间吧! Elasticsearch 拥有众多新功能,助你为自己…...
条件运算符
C中的三目运算符(也称条件运算符,英文:ternary operator)是一种简洁的条件选择语句,语法如下: 条件表达式 ? 表达式1 : 表达式2• 如果“条件表达式”为true,则整个表达式的结果为“表达式1”…...
零基础设计模式——行为型模式 - 责任链模式
第四部分:行为型模式 - 责任链模式 (Chain of Responsibility Pattern) 欢迎来到行为型模式的学习!行为型模式关注对象之间的职责分配、算法封装和对象间的交互。我们将学习的第一个行为型模式是责任链模式。 核心思想:使多个对象都有机会处…...
docker 部署发现spring.profiles.active 问题
报错: org.springframework.boot.context.config.InvalidConfigDataPropertyException: Property spring.profiles.active imported from location class path resource [application-test.yml] is invalid in a profile specific resource [origin: class path re…...
中医有效性探讨
文章目录 西医是如何发展到以生物化学为药理基础的现代医学?传统医学奠基期(远古 - 17 世纪)近代医学转型期(17 世纪 - 19 世纪末)现代医学成熟期(20世纪至今) 中医的源远流长和一脉相承远古至…...
MySQL 8.0 事务全面讲解
以下是一个结合两次回答的 MySQL 8.0 事务全面讲解,涵盖了事务的核心概念、操作示例、失败回滚、隔离级别、事务性 DDL 和 XA 事务等内容,并修正了查看隔离级别的命令。 MySQL 8.0 事务全面讲解 一、事务的核心概念(ACID) 事务是…...
CVPR2025重磅突破:AnomalyAny框架实现单样本生成逼真异常数据,破解视觉检测瓶颈!
本文介绍了一种名为AnomalyAny的创新框架,该方法利用Stable Diffusion的强大生成能力,仅需单个正常样本和文本描述,即可生成逼真且多样化的异常样本,有效解决了视觉异常检测中异常样本稀缺的难题,为工业质检、医疗影像…...
GraphQL 实战篇:Apollo Client 配置与缓存
GraphQL 实战篇:Apollo Client 配置与缓存 上一篇:GraphQL 入门篇:基础查询语法 依旧和上一篇的笔记一样,主实操,没啥过多的细节讲解,代码具体在: https://github.com/GoldenaArcher/graphql…...
面试高频问题
文章目录 🚀 消息队列核心技术揭秘:从入门到秒杀面试官1️⃣ Kafka为何能"吞云吐雾"?性能背后的秘密1.1 顺序写入与零拷贝:性能的双引擎1.2 分区并行:数据的"八车道高速公路"1.3 页缓存与批量处理…...
