热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Linux信号量操作详解:sem_init,sem_wait,sem_post,sem_destroy

本文详细介绍了Linux系统中信号量的相关函数,包括sem_init、sem_wait、sem_post和sem_destroy,解释了它们的功能和使用方法,并提供了示例代码。

信号量基础与操作

信号量是一种用于多线程或多进程环境下的同步机制,主要通过一组API来实现资源的访问控制。本文将详细介绍几个常用的信号量操作函数:sem_init、sem_wait、sem_post和sem_destroy。

sem_init

函数sem_init用于初始化一个信号量,其原型为:

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

参数说明:

  • sem: 指向信号量结构的指针。
  • pshared: 如果设置为0,信号量仅在当前进程中所有线程间共享;若非0,则信号量可在多个进程间共享。
  • value: 信号量的初始值。

例如,初始化一个初始值为0的本地信号量:

sem_t my_semaphore;
int result = sem_init(&my_semaphore, 0, 0);
if (result != 0) {
perror("Semaphore initialization failed");
}

sem_wait

函数sem_wait用于阻塞当前线程,直到信号量的值大于0,然后将其值减1。这通常用于确保在访问临界资源之前,资源可用。其原型为:

int sem_wait(sem_t *sem);

示例:

sem_wait(&my_semaphore);
// 访问临界资源
sem_post(&my_semaphore);

sem_post

函数sem_post用于增加信号量的值。如果当前有线程因信号量值为0而阻塞,则其中一个线程会被唤醒。其原型为:

int sem_post(sem_t *sem);

示例:

sem_post(&my_semaphore);

sem_destroy

函数sem_destroy用于销毁一个信号量,释放相关资源。在调用此函数前,应确保没有线程正在等待该信号量。其原型为:

int sem_destroy(sem_t *sem);

示例:

sem_destroy(&my_semaphore);

完整示例

以下是一个完整的示例,展示了如何使用上述信号量操作函数来控制两个线程之间的同步:

#include 
#include
#include
#include
#include
#include

sem_t bin_sem;

void *thread_function1(void *arg) {
printf("thread_function1--------------sem_wait\n");
sem_wait(&bin_sem);
printf("sem_wait\n");
while (1) {}
}

void *thread_function2(void *arg) {
printf("thread_function2--------------sem_post\n");
sem_post(&bin_sem);
printf("sem_post\n");
while (1) {}
}

int main() {
int res;
pthread_t a_thread;
void *thread_result;

res = sem_init(&bin_sem, 0, 0);
if (res != 0) {
perror("Semaphore initialization failed");
}
printf("sem_init\n");
res = pthread_create(&a_thread, NULL, thread_function1, NULL);
if (res != 0) {
perror("Thread creation failure");
}
printf("thread_function1\n");
sleep(5);
printf("sleep\n");
res = pthread_create(&a_thread, NULL, thread_function2, NULL);
if (res != 0) {
perror("Thread creation failure");
}
while (1) {}
}

以上代码创建了两个线程,其中一个线程调用sem_wait阻塞等待,另一个线程调用sem_post来释放信号量,从而唤醒阻塞的线程。

转自:原文链接


推荐阅读
  • 本文详细介绍了如何在Linux系统上安装和配置Smokeping,以实现对网络链路质量的实时监控。通过详细的步骤和必要的依赖包安装,确保用户能够顺利完成部署并优化其网络性能监控。 ... [详细]
  • 本文详细介绍了 Dockerfile 的编写方法及其在网络配置中的应用,涵盖基础指令、镜像构建与发布流程,并深入探讨了 Docker 的默认网络、容器互联及自定义网络的实现。 ... [详细]
  • 本文深入探讨 MyBatis 中动态 SQL 的使用方法,包括 if/where、trim 自定义字符串截取规则、choose 分支选择、封装查询和修改条件的 where/set 标签、批量处理的 foreach 标签以及内置参数和 bind 的用法。 ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • 扫描线三巨头 hdu1928hdu 1255  hdu 1542 [POJ 1151]
    学习链接:http:blog.csdn.netlwt36articledetails48908031学习扫描线主要学习的是一种扫描的思想,后期可以求解很 ... [详细]
  • 本文探讨了如何在给定整数N的情况下,找到两个不同的整数a和b,使得它们的和最大,并且满足特定的数学条件。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • PyCharm下载与安装指南
    本文详细介绍如何从官方渠道下载并安装PyCharm集成开发环境(IDE),涵盖Windows、macOS和Linux系统,同时提供详细的安装步骤及配置建议。 ... [详细]
  • 本文详细介绍如何使用Python进行配置文件的读写操作,涵盖常见的配置文件格式(如INI、JSON、TOML和YAML),并提供具体的代码示例。 ... [详细]
  • Java 中 Writer flush()方法,示例 ... [详细]
  • 技术分享:从动态网站提取站点密钥的解决方案
    本文探讨了如何从动态网站中提取站点密钥,特别是针对验证码(reCAPTCHA)的处理方法。通过结合Selenium和requests库,提供了详细的代码示例和优化建议。 ... [详细]
  • 题目描述:给定n个半开区间[a, b),要求使用两个互不重叠的记录器,求最多可以记录多少个区间。解决方案采用贪心算法,通过排序和遍历实现最优解。 ... [详细]
  • 在前两篇文章中,我们探讨了 ControllerDescriptor 和 ActionDescriptor 这两个描述对象,分别对应控制器和操作方法。本文将基于 MVC3 源码进一步分析 ParameterDescriptor,即用于描述 Action 方法参数的对象,并详细介绍其工作原理。 ... [详细]
  • 题目Link题目学习link1题目学习link2题目学习link3%%%受益匪浅!-----&# ... [详细]
author-avatar
大布丁
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有