Linux高级编程:进程间的通信(二)、IPC

回顾

共7种方式:

古老的进程间通信方式:

        管道:

                无名管道

                有名管道

        信号

系统V IPC进程对象

        共享内存

        消息队列

        信号量集

socket通信  //网络

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

无名管道

        pipe()

特点:

        用于亲缘关系进程间

     (继承相关的文件描述符 -- 对应实际同一个管道)

管道的特点:

        1.形成的数据流

        2.FIFO(first in first out ) // -- 数据结构的特点 -- 队列(排队)

           FIFO (first in last out)//  栈的特点

        3.管道中的的数据,读取之后,就没有了

管道读写规则:

        a.读端存在,写管道

        b.独断不存在,写管道 --- 股电脑破裂  SIGPIPE 信号 --- 进程收到这信号后会被结束掉

        c.写段存在,读管道 --- 没数据,读操作阻塞

        d.写端变速存在,读管道 -- 没数据,读操作不阻塞, 立即返回  返回0 表示没数据

有名管道:

创建

        mkfifo // 在系统中创建了一个 管道文件的名字

                   // 不同的进程(毫无亲缘关系的进程)可以通过该名字  最终操作到同一个管道进而实现通信

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

信号通信:

 

异步、同步:

用户自定义操作:

signal   

       #include <signal.h>

       typedef void (*sighandler_t)(int);

                        // ===》void (*xx)(int); == void fun(int);
                           ===》xx是 void fun(int) 类型函数的函数指针
                           ===》typedef void(*xx)(int)   sighandler_t;

       sighandler_t signal(int signum, sighandler_t handler);

        功能:

                用于向操作系统注册信号处理函数,以便在特定信号发生时执行相应的操作。

        参数:

                 @signum 是信号编号,它是一个整数,用于指定要处理的信号。

       @handler是一个函数指针,它指向的信号处理函数将在接收到 signum 指定的信号时被                                   调用。       

  • 当handler为SIG_IGN时,表示忽略此信号(需要注意的是,SIGKILLSIGSTOP信号不能被忽略)。
  • 当handler为SIG_DFL时,表示接到此信号后的动作是系统默认动作。
  • 当handler是一个函数地址时,表示当接收到对应编号的signo信号时,执行该函数。

        返回值:

       signal函数的返回值也是一个 sighandler_t 类型的函数指针,它指向之前注册        在 signum 上的信号处理函数。如果注册失败,signal 函数将返回 SIG_ERR

                

练习:

1、自定义处理信号10和12

#include<stdio.h>
#include<unistd.h>
#include<signal.h>void handler_10_12(int sno)
{printf("recieve %d\n",sno);
}int main(int argc, const char *argv[])
{if(signal(SIGUSR1,handler_10_12) == SIG_ERR){perror("signal fail");return -1;}if(signal(SIGUSR2,handler_10_12) == SIG_ERR){perror("signal fail");return -1;}while(1){printf("hello\n");sleep(1);}return 0;
}

2、通过信号处理的方式,回收僵尸态子进程   // 异步处理收尸,不阻碍父进程的操作

#include<stdio.h>
#include<unistd.h>
#include<signal.h>
#include<stdlib.h>
#include <sys/wait.h>void do_wait(int sno)
{wait(0);printf("child wait\n");
}int main(int argc, const char *argv[])
{if(signal(SIGCHLD,do_wait) == SIG_ERR){perror("signal fail");return -1;}pid_t pid = fork();if(pid < 0){perror("fork fail");return -1;}if(pid > 0){int i = 0;while(1){printf("father = %d\n",i++);sleep(1);}}if(pid == 0){int i = 0;while(1){printf("child = %d\n",i++);sleep(1);}}return 0;
}

3、

函数: alarm();  计数  括号内写秒数,到时发送SIGALRM信号  则说明超时1次。

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <signal.h>int count;void handler(int signo)
{printf("---signal---%d\n",++count);;if(count == 3){printf("---timeout---\n");exit(0);}alarm(3);
}int main(int argc, const char *argv[])
{alarm(3);char buf[1024] = {0};signal(SIGALRM,handler);while(1){fgets(buf,sizeof(buf),stdin);count = 0;alarm(3);}return 0;
}

kill   

      #include <sys/types.h>
      #include <signal.h>

      int   kill(pid_t pid, int sig);
            功能:通过该函数可以给pid进程发送信号为sig的系统信号。
            参数:   @pid 要接收信号的进程pid
                          @sig 当前程序要发送的信号编号 《=== kill  -l
            返回值: 成功 0
                            失败  -1;

raise

        int raise(int sig)<==> kill(getpid(),int sig);  
            功能:给进程自己发送sig信号

pause

        int pause(void);
            功能:
                 进程暂停,不再继续执行,除非收到其他信号。

IPC对象:


(*)共享内存 //进程间效率最高的通信方式

shm,sem,msg

    system v :  共享内存  信号量集

    IPC对象操作通用框架:
    0x  ftok
    key值 ==> 申请 ==》读写 ==》关闭 ==》卸载

    
    key值    // 
    申请  -- xxxget  //key <--> ipc对象 
    读写 
    关闭 
    卸载

key值:===》唯一键值
创建方式有三种:

1、IPC_PRIVATE 固定的私有键值,其值等于 0x0
        一般用于有亲缘关系的进程间使用。

2、ftok()创建临时键值。


            #include <sys/types.h>
            #include <sys/ipc.h>
            key_t ftok(const char *pathname, int proj_id);
                    功能:通过该函数可以将pathname指定的路径用来以
                               proj_id生成唯一的临时键值。
                    参数:@pathname 路径+名称===》任意文件,只要不会
                               被删除重建即可。
                               @proj_id  整形的数字,一般用ASCII码的单字符
                                表示与参数1的运算。

                    返回值:成功 返回唯一键值
                                  失败  -1;

申请:

1、申请对象:shmget()
            

            #include <sys/ipc.h>
            #include <sys/shm.h>
                        ps aux|grep a.out
                        shared memory get         IPC_CREAT|0666
            int shmget(key_t key, size_t size, int shmflg);
                    功能:
                          使用唯一键值key向内核提出共享内存使用申请
                    参数:

                          @key   唯一键值
                          @size  要申请的共享内存大小
                          @shmflg 申请的共享内存访问权限,八进制表示
                                  如果是第一个申请,则用IPC_CREAT
                                  如果要检测是否存在,用IPC_EXCL
                    返回值:
                            成功 返回共享内存id,一般用shmid表示
                            失败  -1;

2、映射对象:shmat()
    

        void *shmat(int shmid, const void *shmaddr, int shmflg);
            功能:将指定shmid对应的共享内存映射到本地内存。
            参数:

                     @shmid 要映射的本地内存
                     @shmaddr 本地可用的地址,如果不确定则用NULL,表示
                                由系统自动分配。
                     @shmflg  
                                   0         ,  表示读写
                                   SHM_RDONLY, 只读
            返回值:
                         成功 返回映射的地址,一般等于shmaddr
                         失败 (void*)-1

3、删除对象:shmctl
    

        int shmctl(int shmid, int cmd, struct shmid_ds *buf);
            功能:
                       修改共享内存属性,也可以删除指定的共享内存对象。
            参数:@shmid 要删除的共享内存对象
                       @cmd 
                       @IPC_RMID 删除对象的宏
                       @buff  NULL 表示只删除对象。
            返回值:成功 0
                           失败 -1

练习:

// a
#include<stdio.h>
#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/shm.h>
#include<string.h>
#include<unistd.h>
#include <signal.h>int main(int argc, const char *argv[])
{key_t key = ftok("/",'A');if (key < 0){perror("ftok fail");return -1;}int shmid = shmget(key,1024,IPC_CREAT|0666);if(shmid < 0){perror("shmget fail");return -1;}void *p = shmat(shmid,NULL,0);if(p == (void *)-1){perror("shmat fail");return -1;}pid_t *pid = p;pid_t pid_b = *pid;printf("pid_b = %d\n",pid_b);while(1){fgets(p,1024,stdin);kill(pid_b,10);printf("p = %s\n",(char *)p);if(strncmp((char *)p,"quit",4) == 0){break;}}return 0;
}//b
#include<stdio.h>
#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/shm.h>
#include<string.h>
#include<unistd.h>
#include<signal.h>void handler(int signo)
{}int main(int argc, const char *argv[])
{key_t key = ftok("/",'A');if (key < 0){perror("ftok fail");return -1;}int shmid = shmget(key,1024,IPC_CREAT|0666);if(shmid < 0){perror("shmget fail");return -1;}void *p = shmat(shmid,NULL,0);pid_t *pid = p;*pid = getpid();printf("pid = %d\n",*pid);if(p == (void *)-1){perror("shmat fail");return -1;}signal(SIGUSR1,handler);while(1){	pause();printf("p = %s\n",(char *)p);if(strncmp((char *)p,"quit",4) == 0){break;}}return 0;
}

消息队列 

信号量集

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

卡密交易系统 卡密社区SUP系统源码 分销系统平台 分销商城系统开发

卡密社区SUP系统总控源码主站分销系统功能源码 跟以前的卡盟那种控制端差不多总控可以给别人开通&#xff0c;分销&#xff0c;主站&#xff0c;类似自己做系统商一样&#xff0c;自助发卡&#xff0c;卡密交易系统。 搭建环境Nginx1.22 mysql 5.7 php8.1 rids 7.2 安装方法…

C++ 红黑树

1.红黑树的概念 红黑树&#xff0c;是一种二叉搜索树&#xff0c;但在每个结点上增加一个存储位表示结点的颜色&#xff0c;可以是Red或 Black。 通过对任何一条从根到叶子的路径上各个结点着色方式的限制&#xff0c;红黑树确保没有一条路 径会比其他路径长出俩倍&#xff0c…

网络安全: Kali Linux 使用 docker-compose 部署 openvas

目录 一、实验 1.环境 2.Kali Linux 安装docker与docker-compose 3.Kali Linux 使用docker-compose方式部署 openvas 4. KaliLinux 使用openvas 二、问题 1. 信息安全漏洞库 2.信息安全漏洞共享平台 3.Windows 更新指南与查询 4.CVE 查询 5.docker-compose 如何修改o…

运维随录实战(4)

添加账号并为账号赋予root权限 1,使用root账号添加一个普通账号 adduser test passwd test # 赋予密码 2,赋予root权限 修改/etc/sudoers文件,如果使用vi 命令打开提示仅只读,则使用 visudo命令打开 在root下面添加一行“test ALL=(ALL) ALL”,如下所示 3,将test账…

Python的http模块requests

模块简介&#xff1a; requests 库是一个 python中比较有名的 http请求的库&#xff0c;能处理 get,post,put,delete 等 restful请求&#xff0c;能设置 header&#xff0c;cookie,session 等操作&#xff0c;也是作为爬虫的基础库&#xff0c;它目前还不能异步请求,如果要支持…

C盘空间爆红排查

缘由&#xff1a; 大概率pagefile.sys占用空间非常大&#xff0c;本来我C盘很空的 虚拟内存太大导致 默认windows貌似会把虚拟内存设置成跟物理内存一样大&#xff0c;甚至还会大一点&#xff0c;并且单独存放到C盘&#xff0c;所以这就是为什么我升级内存条后&#xff08;共…

数据库管理-第158期 Oracle Vector DB AI-09(20240304)

数据库管理158期 2024-03-04 数据库管理-第158期 Oracle Vector DB & AI-09&#xff08;20240304&#xff09;1 创建示例表2 添加过滤条件的向量近似查询示例1示例2示例3示例4示例5示例6示例7 总结 数据库管理-第158期 Oracle Vector DB & AI-09&#xff08;20240304&a…

软件工程顶会——ICSE '24 论文清单、摘要

1、A Comprehensive Study of Learning-based Android Malware Detectors under Challenging Environments 近年来&#xff0c;学习型Android恶意软件检测器不断增多。这些检测器可以分为三种类型&#xff1a;基于字符串、基于图像和基于图形。它们大多在理想情况下取得了良好的…

前端监控为什么采用GIF图片做埋点?

一、什么是埋点监控 前端监控是开发人员用来跟踪和维护应用程序表现层的运行状况的过程和工具。它主要包括三种类型&#xff1a;数据监控、性能监控和异常监控。 1、数据监控 主要是为了收集跟用户相关的数据&#xff0c;例如用户设备类型、浏览器版本、页面浏览量&#xff08;…

智奇科技工业 Linux 屏更新开机logo

智奇科技工业 Linux 屏更新开机logo 简介制作logo.img文件1、转换格式得到logo.bmp2、使用Linux命令生成img文件 制作rootfs.img文件替换rootfs.img中的logo 生成update.img固件附件 简介 智奇科技的 Linux 屏刷开机logo必须刷img镜像文件&#xff0c;比较复杂。 制作logo.i…

ABAP - 增强:一代增强User exit

一代增强是基于源代码的增强&#xff0c;一般是名字UserExit_开头空代码的子例程&#xff0c;所以一代增强的别称用户出口。需要修改SAP标准标准代码集中在名称倒数第二位为’Z‘的include程序里面。所有的全局数据可用那么该如何找到一代增强呢&#xff1f;以销售订单为例&…

突破编程_前端_JS编程实例(简单树结构组件)

1 开发目标 实现如下简单树结构组件&#xff1a; 再点击树节点后&#xff0c;会调用客户端传入的回调函数&#xff1a; 2 详细需求 简单树结构组件需根据客户端提供的参数创建&#xff0c;具备动态构建树形结构节点、选项卡切换及自定义内容显示等功能&#xff1a; &#xf…