Linux进程与线程之五

2024-05-19 10:18
文章标签 linux 线程 进程 之五

本文主要是介绍Linux进程与线程之五,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

                               每日一结

一 共享内存 :内核空间预留出来的一块内存,用于进程间通信 

(1)int shmget(key_t key, size_t size, int shmflg);

功能:获取共享内存段的ID 

参数:

@key    IPC_PRIVATE  或 ftok() 

@size   申请的共享内存段大小 [4k的倍数]

@shmflg IPC_CREAT | 0666 或 IPC_CREAT | IPC_EXCL 

返回值:

成功返回ID,失败返回-1 

(2)void *shmat(int shmid, const void *shmaddr, int shmflg);

功能:映射共享内存到用户空间 

参数:

@shmid    共享内存段ID 

@shmaddr  NULL:系统自动完成映射 

@shmflg   SHM_RDONLY:只读   0:读写

返回值:

成功返回映射后的地址,失败返回(void *)-1 

 

练习:A,B通过共享内存通信

B如何知道A已经写了数据?

 

flag    

----------------------------------------------

 0    |  内容

-----------------------------------------------

 

(3)int shmdt(const void *shmaddr);

功能:撤销映射

参数:

@shmaddr  共享内存映射的地址 

(4)int shmctl(int shmid, int cmd, struct shmid_ds *buf);

功能:根据命令控制共享内存 

参数:

@shmid  共享内存段的ID 

@cmd    IPC_STAT[获取属性],IPC_SET[设置属性],IPC_RMID[删除IPC对象]

@buf    保存属性 

返回值:

成功返回0,失败返回 -1 

 

-----------------------------------------------------------

注意:当我们调用shmctl删除共享内存的时候,并不会立即删除。只有当共享内存映射次数为0,才会删除共享内存对象

代码示例如下:

 

关于Linux环境进程间通信的参考资料:

共享内存:
上:http://www.ibm.com/developerworks/cn/linux/l-ipc/part5/index1.html
下:http://www.ibm.com/developerworks/cn/linux/l-ipc/part5/index2.html

 

 

-----------------------------------------------------------

二 信号灯集

POSIX 线程中的同步用的是无名信号量 

进程间的同步使用的是IPC 对象[信号灯集]

信号灯集:信号灯集合,每一个信号灯都可以用来表示一类资源,其值表示资源的个数

(1)创建信号灯集

int semget(key_t key, int nsems, int semflg);

参数:

@key    IPC_PRIVATE , ftok() 

@nsems  信号灯集中信号灯的个数 

@semflg IPC_CREAT | 0666,IPC_CREAT | IPC_EXCL

返回值:

成功返回ID,失败返回-1 

(2)初始化信号灯集中信号灯的值

 

int semctl(int semid, int semnum, int cmd, ...);

参数:

@semid   信号灯集的ID 

@semnum  信号灯的编号[编号从0开始]

@cmd     SETVAL[设置信号灯的值]  ,GETVAL(获取信号灯的值),IPC_RMID[删除信号灯集]

返回值:

成功返回0,失败返回-1 

 

思考:将信号灯集中的1号信号灯初始化为1?

 

union semun {

int val; /* Value for SETVAL */

struct semid_ds *buf; /* Buffer for IPC_STAT, IPC_SET */

unsigned short  *array; /* Array for GETALL, SETALL */

struct seminfo  *__buf; /* Buffer for IPC_INFO

   (Linux-specific) */

};

 

void init_sem_value(int sem_id,int sem_num,int value)

{

union semun sem_val;

sem_val.val = value;

if(semctl(sem_id,sem_num,SETVAL,sem_val) < 0)

{

...

}

 

return ;

}

 

(3)PV操作 

int semop(int semid, struct sembuf *sops, unsigned nsops);

功能:完成PV操作  

参数:

@semid  信号灯集的ID 

@sops   操作方式结构体首地址 

@nsops  操作信号灯的个数 

返回值:

成功返回0,失败返回-1

 

struct sembuf 

{

unsigned short sem_num;  /* semaphore number */

short   sem_op;   /* semaphore operation  */

short   sem_flg;  /* operation flags */

};

 

sem_op : 

<1>0   等待信号灯的值变成0 

<2>1   释放资源,V操作 

<3>-1  申请资源,P操作 

 

sem_flg:

0           : 阻塞方式 

IPC_NOWAIT  : 非阻塞方式调用 

SEM_UNDO    : 进程结束的时候,它申请的资源自动释放

 

-----------------------------------------------------------------

void P(int sem_id,int sem_num)

{

struct sembuf sem;

 

sem.sem_num = sem_num;

sem.sem_op  = -1;

sem.sem_flg = 0;

if(semop(sem_id,&sem,1) < 0)

{

....

}

}

 

void V(int sem_id,int sem_num)

{

struct sembuf sem;

 

sem.sem_num = sem_num;

sem.sem_op  = 1;

sem.sem_flg = 0;

if(semop(sem_id,&sem,1) < 0)

{

....

}

}

 

关于Linux环境进程间通信的参考资料

信号灯:
http://www.ibm.com/developerworks/cn/linux/l-ipc/part4/

 

练习:

A,B通过信号灯集同步对共享内存操作 

 

让创建信号灯集的进程,初始化信号灯的值 ,如果信号灯集已经存在则不初始化

 

sem_id = semget(key,2,IPC_CREAT | IPC_EXCL | 0666);

if(sem_id < 0)

{

sem_id = semget(key,2,IPC_CREAT | 0666);

}

 

参考代码如下:

sem.c




 


shm_write.c



 

 

 

 

shm_read.c



 

注意:shm_read.c 和 shm_write.c 都需要单独和sem.c一起编译

亦即:gcc shm_read.c sem.c -o shm_read

      gcc shm_write.c sem.c -o shm_write

 


 

关注微信公众号获取更多资讯

这篇关于Linux进程与线程之五的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/995816

相关文章

linux服务之NIS账户管理服务方式

《linux服务之NIS账户管理服务方式》:本文主要介绍linux服务之NIS账户管理服务方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、所需要的软件二、服务器配置1、安装 NIS 服务2、设定 NIS 的域名 (NIS domain name)3、修改主

Linux实现简易版Shell的代码详解

《Linux实现简易版Shell的代码详解》本篇文章,我们将一起踏上一段有趣的旅程,仿照CentOS–Bash的工作流程,实现一个功能虽然简单,但足以让你深刻理解Shell工作原理的迷你Sh... 目录一、程序流程分析二、代码实现1. 打印命令行提示符2. 获取用户输入的命令行3. 命令行解析4. 执行命令

Python多进程、多线程、协程典型示例解析(最新推荐)

《Python多进程、多线程、协程典型示例解析(最新推荐)》:本文主要介绍Python多进程、多线程、协程典型示例解析(最新推荐),本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定... 目录一、multiprocessing(多进程)1. 模块简介2. 案例详解:并行计算平方和3. 实现逻

C#通过进程调用外部应用的实现示例

《C#通过进程调用外部应用的实现示例》本文主要介绍了C#通过进程调用外部应用的实现示例,以WINFORM应用程序为例,在C#应用程序中调用PYTHON程序,具有一定的参考价值,感兴趣的可以了解一下... 目录窗口程序类进程信息类 系统设置类 以WINFORM应用程序为例,在C#应用程序中调用python程序

ubuntu16.04如何部署dify? 在Linux上安装部署Dify的技巧

《ubuntu16.04如何部署dify?在Linux上安装部署Dify的技巧》随着云计算和容器技术的快速发展,Docker已经成为现代软件开发和部署的重要工具之一,Dify作为一款优秀的云原生应用... Dify 是一个基于 docker 的工作流管理工具,旨在简化机器学习和数据科学领域的多步骤工作流。它

Linux高并发场景下的网络参数调优实战指南

《Linux高并发场景下的网络参数调优实战指南》在高并发网络服务场景中,Linux内核的默认网络参数往往无法满足需求,导致性能瓶颈、连接超时甚至服务崩溃,本文基于真实案例分析,从参数解读、问题诊断到优... 目录一、问题背景:当并发连接遇上性能瓶颈1.1 案例环境1.2 初始参数分析二、深度诊断:连接状态与

Linux系统调试之ltrace工具使用与调试过程

《Linux系统调试之ltrace工具使用与调试过程》:本文主要介绍Linux系统调试之ltrace工具使用与调试过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录一、ltrace 定义与作用二、ltrace 工作原理1. 劫持进程的 PLT/GOT 表2. 重定

Linux区分SSD和机械硬盘的方法总结

《Linux区分SSD和机械硬盘的方法总结》在Linux系统管理中,了解存储设备的类型和特性是至关重要的,不同的存储介质(如固态硬盘SSD和机械硬盘HDD)在性能、可靠性和适用场景上有着显著差异,本文... 目录一、lsblk 命令简介基本用法二、识别磁盘类型的关键参数:ROTA查询 ROTA 参数ROTA

嵌入式Linux之使用设备树驱动GPIO的实现方式

《嵌入式Linux之使用设备树驱动GPIO的实现方式》:本文主要介绍嵌入式Linux之使用设备树驱动GPIO的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录一、设备树配置1.1 添加 pinctrl 节点1.2 添加 LED 设备节点二、编写驱动程序2.1

嵌入式Linux驱动中的异步通知机制详解

《嵌入式Linux驱动中的异步通知机制详解》:本文主要介绍嵌入式Linux驱动中的异步通知机制,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言一、异步通知的核心概念1. 什么是异步通知2. 异步通知的关键组件二、异步通知的实现原理三、代码示例分析1. 设备结构