早期进程间的通信

news2025/1/10 0:25:25

目录

IO进程(day06)

无名管道

有名管道

信号


IO进程(day06)

无名管道

  1. 原理图

  1. 无名管道的特点

  1. 只能用于有亲缘关系之间的进程
  2. 无名管道可以看成是一种特殊的文件,对于它的读写可以使用文件IO如read、write函数.
  3. 无名管道是基于文件描述符的通信方式。当一个管道建立时,它会创建两个文件描述符fd[0]和fd[1]。其中fd[0]固定用于读管道,而fd[1]固定用于写管道。
  4. 半双工的通信模式,具有固定的读端和写端

单工:只能单方向传递信息(收音机,广播,电视)

半双工:同一时刻只能接收或者发送(对讲机)

全双工:既可以接收消息也可以发送消息(电话,短信)

  1. 读端关闭的话,会导致管道破裂。
  1. 函数接口
int pipe(int fd[2])
功能:创建无名管道
参数:文件描述符 fd[0]:读端  fd[1]:写端
返回值:成功 0
      失败 -1

示例代码:

注意事项

  1. 当管道中没有数据时,读操作会阻塞;管道中没有数据,将写端关闭,读操作立即返回。
  2. 管道中装满(管道的大小:64k),数据写阻塞,一旦有4k空间,写继续,直到写满。

练习:父子进程间实现通信,父进程循环从终端输入,子进程循环打印,输入quit退出

#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <string.h>

int main(int argc, char const *argv[])
{
    char buf[32];
    int fd[2];
    // 创建无名管道
    if (pipe(fd) < 0)
    {
        perror("pipe err\n");
        return -1;
    }
    printf("pipe create successed\n");
    // 创建子进程
    pid_t pid = fork();
    if (pid < 0)
    {
        perror("fork err\n");
        return -1;
    }
    if (pid == 0)
    {
        // 关闭写端
        close(fd[1]);
        // 循环读取数据
        while (1)
        {
            read(fd[0], buf, 32);
            if (!strcmp(buf, "quit"))
            {
                break;
            }
            printf("buf:%s\n", buf);
            memset(buf, 0, 32);
        }
        // 关闭读端
        close(fd[0]);
    }
    else
    {
        // 关闭读端,防止在写入数据的时候读取数据,造成数据不准确
        close(fd[0]);
        // 循环写入
        while (1)
        {
            fgets(buf, 32, stdin);
            if (buf[strlen(buf) - 1] == '\n')
            {
                buf[strlen(buf) - 1] = '\0';
            }
            write(fd[1], buf, strlen(buf));
            // quit退出
            if (!strcmp(buf, "quit"))
            {
                break;
            }
        }
        // 关闭写端
        close(fd[1]);
        // 阻塞等待子进程结束,回收资源
        wait(NULL);
    }
    return 0;
}

有名管道

  1. 特点
  1. 有名管道可以使互不相关的两个进程互相通信。
  2. 有名管道可以通过路径名来指出,并且在文件系统中可见,但内容存放在内存中。
  3. 进程通过文件IO来操作有名管道
  4. 有名管道遵循先进先出规则,不支持如lseek() 操作
  1. 函数接口
int mkfifo(const char *pathname, mode_t mode);
注意:umask会影响文件的权限。
功能:创建有名管道
参数:filename:有名管道文件名
       mode:权限
返回值:成功:0
       失败:-1,并设置errno号

用法:创建有名管道
if(mkfifo("./fifo",0666) < 0)
{
    perror("mkfifo err");
    return -1;
}
  1. 注意事项
  1. 可读可写,管道中如果没有数据会读阻塞。
  2. 只写方式,写阻塞(open阻塞),一直到另一个进程把读打开。
  3. 只读方式,读阻塞(open阻塞),一直到另一个进程把写打开。

练习:通过两个进程实现cp功能

cp1.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>

int main(int argc, char const *argv[])
{
    // 创建有名管道
    if (mkfifo("./fifo", 0666) < 0)
    {
        if (errno != 17)
        {
            perror("mkfifo err");
            return -1;
        }
    }
    printf("mkfifo ok\n");
    // 打开源文件
    int fd_src = open(argv[1], O_RDONLY);
    if (fd_src < 0)
    {
        perror("open src err");
        return -1;
    }
    // 打开管道文件
    int fd_fifo = open("./fifo", O_WRONLY);
    if (fd_fifo < 0)
    {
        perror("open fifo err");
        return -1;
    }
    // 循环写入
    ssize_t s;
    char buf[32];
    // 将源文件的内容读到数组中
    while (s = read(fd_src,buf,32))
    {
        // 将数组中的内容写入管道文件
        write(fd_fifo,buf,s);
    }
    printf("copy finished\n");
    close(fd_src);
    close(fd_fifo);
    // 关闭文件描述符。
    return 0;
}

cp2.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>

int main(int argc, char const *argv[])
{
    // 创建有名管道
    if (mkfifo("./fifo", 0666) < 0)
    {
        if (errno != 17)
        {
            perror("mkfifo err");
            return -1;
        }
    }
    printf("mkfifo ok\n");
    // 打开源文件
    int fd_dest = open(argv[1], O_WRONLY | O_TRUNC | O_CREAT,0666);
    if (fd_dest < 0)
    {
        perror("open dest err");
        return -1;
    }
    // 打开管道文件
    int fd_fifo = open("./fifo", O_RDONLY);
    if (fd_fifo < 0)
    {
        perror("open fifo err");
        return -1;
    }
    // 循环写入
    ssize_t s;
    char buf[32];
    while (s = read(fd_fifo,buf,32))
    {
        // 将数组中的内容写入管道文件
        write(fd_dest,buf,s);
    }
    close(fd_dest);
    close(fd_fifo);
    // 关闭文件描述符。
    return 0;
}
  1. 有名管道和无名管道的区别

无名管道

有名管道

使用场景

具有亲缘关系的进程间

不相关的两个进程可以使用

特点

一种特殊的文件,通过文件IO进行读写,有固定的读fd[0],和写fd[1],不支持lseek,遵循先进先出

在文件系统中会存在,数据存放在内核空间,通过文件IO进行读写,不支持lseek,遵循先进先出

函数

pipe

mkfifo

读写特性

当管道中没有数据时,读阻塞,写端关闭,读操作立即返回,管道写满时,写阻塞,读端关闭,管道破裂

可读可写,如果管道中没有数据看,读阻塞

只写方式打开,写阻塞,一直到另一个进程读打开。

只读方式打开,读阻塞,一直到另一个进程写打开。

信号

  1. 信号的概念

  1. 信号是在软件层次上对中断机制的一种模拟,是一种异步的通信方式。
  2. 信号可以直接进行用户空间进程和内核进程之间的交互,内核进程也可以利用它来通知用户空间进程发生了哪些系统事件。
  3. 如果该进程当前并未处于执行态,则该信号就由内核保存起来,直到该进程恢复执行再传递给它;如果一个信号被进程设置为阻塞,则该信号的传递被延迟,直到其阻塞被取消时才被传递给进程。
  1. 信号的响应方式

  1. 忽略信号:对信号不进行任何处理,但是有两个信号不能忽略:SIGKILL 和 SIGSTOP
  2. 捕捉信号:定义信号处理函数,当信号发生时,执行相应的处理函数。但是有两个信号不能被捕捉:即SIGKILL及SIGSTOP。
  3. 执行缺省操作:linux对每种信号都规定了默认操作
SIGINT:结束进程,对应快捷方式ctrl+c
SIGQUIT:退出信号,对应快捷方式ctrl+\
SIGKILL:结束进程,不能被忽略不能被捕捉
SIGALRM:闹钟信号,alarm函数设置定时,当到设定的时间时,内核会向进程发送此信号结束进程。
SIGTERM:结束终端进程,kill 使用时不加数字默认是此信号
SIGCHLD:子进程状态改变时给父进程发的信号
SIGSTOP:暂停进程,不能被忽略不能被捕捉
SIGTSTP:暂停信号,对应快捷方式ctrl+z
  1. 信号的种类

在Linux中,信号被分为不可靠信号和可靠信号,一共64种,可以通过kill -l命令来查看

  • 不可靠信号:也称为非实时信号,不支持排队,信号可能会丢失,比如发送多次相同的信号,进程只能收到一次,信号值取值区间为1~31
  • 可靠信号:也称为实时信号,支持排队,信号不会丢失,发多少次,就可以收到多少次,信号值取值区间为32~64

信号产生的方式有如下几种:

  • 对于前台进程,用户可以输入特殊终端字符来发送,比如输入Ctrl+C
  • 系统异常,比如浮点异常和非法内存段访问(段错误)
  • 系统状态变化,比如alarm定时器到期时将引起SIGALRM信号
  • 在终端运行kill命令或在程序中调用kill函数 kill -9 PID:杀死进程
  1. 函数接口

int kill(pid_t pid, int sig);
功能:信号发送
参数:pid:指定进程
   sig:要发送的信号
返回值:成功 0     
       失败 -1

int raise(int sig);
功能:进程向自己发送信号
参数:sig:信号
返回值:成功 0   
      失败 -1

int pause(void);
功能:用于将调用进程挂起,直到收到信号为止。

unsigned int alarm(unsigned int seconds)
功能:在进程中设置一个定时器
参数:seconds:定时时间,单位为秒
返回值:如果调用此alarm()前,进程中已经设置了闹钟时间,则
返回上一个闹钟时间的剩余时间,否则返回0。
注意:一个进程只能有一个闹钟时间。如果在调用alarm时
已设置过闹钟时间,则之前的闹钟时间被新值所代替

信号处理函数

sighandler_t signal(int signum, sighandler_t handler);
功能:信号处理函数
参数:signum:要处理的信号
      handler:信号处理方式
            SIG_IGN:忽略信号
            SIG_DFL:执行默认操作
           handler:捕捉信号  void handler(int sig){} //函数名可以自定义
返回值:成功:设置之前的信号处理方式
      失败:-1
      
用法:
//忽略SIGINT信号
signal(SIGINT,SIG_IGN)
//缺省
signal(2,SIG_DFL)

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux系统块存储子系统分析记录

1 Linux存储栈 通过网址Linux Storage Stack Diagram - Thomas-Krenn-Wiki-en&#xff0c;可以获取多个linux内核版本下的存储栈概略图&#xff0c;下面是kernel-4.0的存储栈概略图&#xff1a; 2 存储接口、传输速度 和 协议 2.1 硬盘 《深入浅出SSD&#xff1a;固态存储核心…

Python爬虫入门篇!

毕设是做爬虫相关的&#xff0c;本来想的是用java写&#xff0c;也写了几个爬虫&#xff0c;其中一个是爬网易云音乐的用户信息&#xff0c;爬了大概100多万&#xff0c;效果不是太满意。之前听说Python这方面比较强&#xff0c;就想用Python试试&#xff0c;之前也没用过Pytho…

从0开始搭建一个生产级SpringBoot2.0.X项目(三)SpringBoot接口统一返回和全局异常处理

前言 最近有个想法想整理一个内容比较完整springboot项目初始化Demo。 SpringBoot接口统一返回和全局异常处理&#xff0c;使用ControllerAdvice ExceptionHandler 的组合来实现。 一、pom文件新增依赖 <dependency><groupId>com.alibaba</groupId><ar…

【MySQL】实战篇—项目需求分析:ER图的绘制与关系模型设计

在软件开发中&#xff0c;数据库是信息系统的核心部分&#xff0c;合理的数据库设计能够显著提高系统的性能和可维护性。 ER图&#xff08;实体-关系图&#xff09;是数据库设计的重要工具&#xff0c;它通过图形化的方式描述了数据实体及其相互关系&#xff0c;帮助开发者和设…

输入整数n,求,i从1到n的和

// 第一题&#xff0c;输入整数n&#xff0c;求&#xff0c;i从1到n的和 #define _CRT_SECURE_NO_WARNINGS #include<stdio.h> int main() {int n 0;printf("请输入一个整数n:");scanf("%d", &n);int i 0;int j 0;for (j 1; j < n; j){i…

频率限制:WAF保护网站免受恶意攻击的关键功能

频率限制&#xff08;Rate Limiting&#xff09;是一项有效的安全措施&#xff0c;用于控制每个 IP 地址的访问速率&#xff0c;以防止恶意用户利用大量请求对网站进行攻击&#xff0c;例如防止 CC 攻击等。频率限制不仅能保护网站资源&#xff0c;还能提升服务的稳定性。 下面…

C++基础: string(3)

文章目录 1. 两道题目1. [387. 字符串中的第一个唯一字符 - 力扣&#xff08;LeetCode&#xff09;](https://leetcode.cn/problems/first-unique-character-in-a-string/description/)2.[415. 字符串相加 - 力扣&#xff08;LeetCode&#xff09;](https://leetcode.cn/proble…

建设NFS服务器并实现文件共享

关闭防火墙和s0 systemctl stop firewalld setenforce 0 安装NFS yum install nfs-utils -y 新建共享目录并设置权限 echo "hello" > /nfs/shared/test1 chmod -Rf 777 /nfs/shared/ 配置服务端的NFS配置文件 vim /etc/exports /nfs/shared *(ro) 启动…

【Java】方法的使用 —— 语法要求、方法的重载和签名、方法递归

目录 1. 方法基础知识 1.1 方法的概念 1.2 语法格式 * 注意事项【与C不同】 1.3 return —— 返回值的严格检查【比C语言严格】 2. 形参与实参的关系 3. 方法重载 3.1 什么是方法重载&#xff1f;为什么要方法重载&#xff1f; 3.2 方法重载的规则 4. 方法签名 5. 递…

Chrome和Firefox如何保护用户的浏览数据

在当今数字化时代&#xff0c;保护用户的浏览数据变得尤为重要。浏览器作为我们日常上网的主要工具&#xff0c;其安全性直接关系到个人信息的保密性。本文将详细介绍Chrome和Firefox这两款主流浏览器如何通过一系列功能来保护用户的浏览数据。&#xff08;本文由https://chrom…

20241030在荣品PRO-RK3566开发板的适配Rockchip原厂的buildroot的时候配置DTS中的电源域

20241030在荣品PRO-RK3566开发板的适配Rockchip原厂的buildroot的时候配置DTS中的电源域 2024/10/30 17:38 请问 RK3566开发板上的 电源配置 和 DTS文件是如何对应的&#xff1f; 底板原理图 PRO-RK3566-B-20210329原理图.pdf vccio4-supply 是1.8V。 对不上呀&#xff1f; Z:…

Java 内部类(13/30)

目录 Java 内部类 1. 内部类的概念和类型 1.1 成员内部类 1.2 局部内部类 1.3 匿名内部类 1.4 静态内部类 2. 内部类的用途和优势 3. 内部类的注意事项 总结与后续 Java 内部类 内部类&#xff08;Inner Class&#xff09;是定义在另一个类内部的类。在 Java 中&…

c++编解码封装

多态版编解码 对服务器和客户端的结构体进行序列化然后对数据进行反序列化 案例分析 代码demo Codec.h #pragma once #include <iostream>class Codec { public:Codec();virtual std::string encodeMsg();//string是标准库的string类virtual void* decodeMsg();virtu…

WPF拖拽交互全攻略及实现自定义拖拽控件及数据交换技巧解析

目录 1. 基本概念2 . 实现拖拽功能概述需要要实现基本的拖放&#xff0c;完成以下任务&#xff1a;其他操作 示例3.1 设置拖拽源&#xff0c;拖拽开始3.2 设置拖拽效果DragDropEffects 3.3 设置放置目标&#xff0c;处理拖拽数据拖拽输入DragEnter事件DragOver事件拖拽离开Drag…

【jvm】为什么Xms和Xmx的值通常设置为相同的?

目录 1. 说明2. 避免性能开销3. 提升稳定性4. 简化配置5. 优化垃圾收集6. 获取参数6.1 代码示例6.2 结果示例 1. 说明 1.-Xms 和 -Xmx 参数分别用于设置堆内存的初始大小&#xff08;最小值&#xff09;和最大大小。2.在开发环境中&#xff0c;开发人员可能希望快速启动应用程…

c++提示函数可能不安全This function or variable may be unsafe

现象 编译c代码时&#xff0c;经常会出现如下提示&#xff0c;大约有几种情况&#xff1a; 现象一 warning C4996: strcpy: This function or variable may be unsafe. Consider using strcpy_s instead. To disable deprecation, use _CRT_SECURE_NO_WARNINGS. See online …

苹果ANE引擎以及使用编程方法介绍

苹果ANE引擎介绍 大多数新款iPhone和iPad都配备了一个神经网络引擎&#xff08;Neural Engine&#xff09;&#xff0c;这是一种特殊的处理器&#xff0c;能让机器学习模型的运行速度变得非常快。然而&#xff0c;关于这个处理器具体如何工作&#xff0c;公众所知的信息并不多…

Python | Leetcode Python题解之第516题最长回文子序列

题目&#xff1a; 题解&#xff1a; class Solution:def longestPalindromeSubseq(self, s: str) -> int:n len(s)dp [[0] * n for _ in range(n)]for i in range(n - 1, -1, -1):dp[i][i] 1for j in range(i 1, n):if s[i] s[j]:dp[i][j] dp[i 1][j - 1] 2else:dp…

网络编程 UDP编程 Linux环境 C语言实现

UDP编程 1. 一般UDP编程 UDP传输特点&#xff1a;非面向连接、不可靠的、无序的 报式传输 支持组播和广播 UDP应用数据最大长度建议&#xff1a;MTU(以太网分组数据的最大长度)1500 - 20(IP头) - 8(UDP头) 1472Bytes 客户端&#xff1a;支持两种形式的代码编写: 1. 不定向…

高压线路覆冰厚度测量,输电线路微波覆冰监测装置守护电网安全

随着北方地区正式步入冬季&#xff0c;气温急剧下降&#xff0c;高压线路覆冰现象日益严重&#xff0c;给电网安全带来了前所未有的挑战。近日&#xff0c;在东北某省&#xff0c;由于连续低温天气&#xff0c;多条高压线路遭受了严重的覆冰侵袭&#xff0c;这不仅极大地加重了…