0

0

【Linux】命名管道的妙用:实现进程控制与实时字符交互

絕刀狂花

絕刀狂花

发布时间:2025-08-13 10:50:19

|

776人浏览过

|

来源于php中文网

原创

1. 进程池

在匿名管道学习后,我们已经实现了一个基于匿名管道的简单进程控制,现在我们学习了命名管道来试试用命名管道来实现进程控制吧。 正在匿名管道时,我们已经实现了进程的控制,命名管道只要在其基础上进行一些修改就可以了。

2. 进程池的功能2.1 可被执行的任务

进程池是为了实现对进程的控制,通过父进程对众多子进程的管理,实现高效的进程执行任务。 我们可以先虚构一些任务出来,后续的子进程就来执行这些任务:

打印日志将数据插入数据库请求网络 可以用这些任务可以写成单独的函数,然后用一个类来管理这些任务。 类的成员变量: 利用哈希表来存储这些任务 类的功能有:展示可执行的任务执行任务 具体的代码如下:代码语言:javascript代码运行次数:0运行复制
typedef void(*task_t)(void);/*函数指针*///任务集void printLog(){    std::cout<<"PID:"< st_;public:    Task()    {        st_ = {{"打印日志",printLog},{"插入数据",MySQLInsert},{"请求网络",NetRequest}};    }    void showTask()/*展示任务*/    {        std::cout<<"可选任务:"<2.2 进程控制(重点)

进程控制是进程池的重点、核心。我们实现进程池也就是为了执行对进程的控制。 如何控制进程呢?

创建进程展示空闲子进程子进程等待任务的发配父进程为子进程发配任务杀死/回收进程2.2.1 子进程类

为了实现这些功能,同样我们用一个类来封装它们。除此之外,还记得我们在实现基于匿名管道的进程控制吗?我们还需要一个类来描述子进程。因为我们需要知道父进程的每个写端的文件描述符和哪个子进程通过命名管道建立起来联系。 所以子进程类具有的属性就有了:

子进程的PID写端的文件描述符编号名字 其中第3和第4个属性是为了方便父进程管理而加入的,后续进程控制类需要用数组将各个子进程存储起来。 子进程类代码如下:代码语言:javascript代码运行次数:0运行复制
class SubProcesses{private:    pid_t id_;    int processNum_;    int wfd_;    std::string name_;    static int cnt_;public:    SubProcesses(pid_t id,int wfd)    :id_(id),processNum_(cnt_++),wfd_(wfd)    {        char buff[SIZE]{0};        snprintf(buff,SIZE,"Process %d | pid:wfd [%d:%d]",processNum_,id_,wfd_);        name_ = buff;    }    pid_t getId()    {        return id_;    }    std::string getName()    {        return name_;    }    int getWfd()    {        return wfd_;    }};int SubProcesses::cnt_ = 1;
2.2 进程控制

那么进程类会具有什么属性呢?我们需要将创建出子进程进行管理可以用一个

vecotr
来对其进行管理,还需要知道我们需要创建的子进程数量,因为是基于命名管道的进程控制,为此我们还必须对个个进程管道进行管理,同样用
vector
将它们装起来。除此之外在来个权限掩码也可以。 如此进程控制类的成员变量:

管理着子进程的
vector
类。管理着命名管道的
vector
类。记录需要创建的子进程数目。权限掩码。 下面开始实现函数:2.2.2 创建子进程

创建子进程,创建子进程同样还是用到

fork
函数,不同的是不在需要
pipe
函数来创建管道了。现在已经变成了
mkfifo
函数。 和匿名管道那一样,我们要记得关闭子进程的写端,尽管子进程并没有主动打开写端,但是因为子进程继承父进程的缘故,依然会吧父进程的写端给继承过了的。一点定是要关闭的。

代码语言:javascript代码运行次数:0运行复制
void createProcess(){std::vector vfd;/*关闭子进程的写端*/for(int i = 1;i<=subProcessNum_;++i){std::string fifoName = "fifo-"+std::to_string(i); int ret = mkfifo(fifoName.c_str(),mode_);/*创建命名管道*/if(ret == -1){perror("mkfifo");exit(1);}namePipe_.push_back(fifoName);/*存入命名管道*/pid_t id = fork();/*创建子进程*/if(id < 0){perror("fork");exit(1);}else if(id == 0){//子进程for(auto&fd:vfd){close(fd);}int rfd = open(fifoName.c_str(),O_RDONLY);if(rfd == -1){perror("open");exit(1);}//开始等待父进程的指令waitTask(rfd); /*子进程等待任务委派*/close(rfd);exit(0);}//父进程int wfd = open(fifoName.c_str(),O_WRONLY);if(wfd == -1){perror("open");exit(1);}subProcess_.push_back({id,wfd});/*存入子进程信息*/fifoName.push_back(wfd);}}
2.2.3 子进程等待任务

死循环等待任务,当写端关闭再退出。

代码语言:javascript代码运行次数:0运行复制
void waitTask(int rfd)/*子进程等待任务委派*/{while(true){char buff[SIZE];int n = read(rfd,buff,SIZE);if(n == -1){perror("read");exit(1);}else if(n>0){Task().curTask((std::string)buff);}else if(n == 0){/*证明写端没有写消息了*/std::cout<<"写端已经关闭,读端也即将关闭!"<2.2.4 展示空闲进程代码语言:javascript代码运行次数:0运行复制
/*展示可选进程*/    void showProcess()    {        std::cout<<"目前可用进程有:"<2.2.5 分配任务给子进程

该函数的功能就分配任务,由用户自己选择子进程来执行任务。

英特尔AI工具
英特尔AI工具

英特尔AI与机器学习解决方案

下载
代码语言:javascript代码运行次数:0运行复制
/*下达任务给子进程*/void sendTask(){std::cout<<"------------------------"<>input;   }while(input<0||input>=subProcessNum_);Task().showTask();std::string taskName;std::cout<<"请选择任务#";std::cin>>taskName;if(taskName == "exit"){break;}std::cout<<"选择进程-> "<2.2.6 回收子进程

回收子进程的同时也不能忘记关闭文件描述符和关闭命名管道

代码语言:javascript代码运行次数:0运行复制
/*关闭写端、删除文件、等待子进程退出*/    void waitProcess()    {        for(int i = 0;i代码代码语言:javascript代码运行次数:0运行复制
#include #include #include #include #include #include #include #include #include #include /** * 该文件为进程池的公共代码段 * 进程池为利用命名管道实现的,简称命名管道进程池 * 代码的主要功能包括:创建进程,控制进程实现委派给它的任务,等待进程 */#define SIZE 1024typedef void(*task_t)(void);//任务集void printLog(){    std::cout<<"PID:"< st_;public:    Task()    {        st_ = {{"打印日志",printLog},{"插入数据",MySQLInsert},{"请求网络",NetRequest}};    }    void showTask()/*展示任务*/    {        std::cout<<"可选任务:"< subProcess_;/*子进程信息表*/    std::vector namePipe_;/*命名管道信息表*/    int subProcessNum_ ;/*需要创建的子进程数目*/    mode_t mode_ ;public:    ProcessCtrl(int subProcessNum = 3,mode_t mode = 0666)    :subProcessNum_(subProcessNum),mode_(mode)    {createProcess();/*开始创建子进程*/}    void createProcess()    {        std::vector vfd;/*关闭子进程的写端*/        for(int i = 1;i<=subProcessNum_;++i)        {            std::string fifoName = "fifo-"+std::to_string(i);             int ret = mkfifo(fifoName.c_str(),mode_);/*创建命名管道*/            if(ret == -1)            {                perror("mkfifo");                exit(1);            }            namePipe_.push_back(fifoName);/*存入命名管道*/            pid_t id = fork();/*创建子进程*/            if(id < 0)            {                perror("fork");                exit(1);            }            else if(id == 0)            {                //子进程                for(auto&fd:vfd)                {                    close(fd);                }                int rfd = open(fifoName.c_str(),O_RDONLY);                if(rfd == -1)                {                    perror("open");                    exit(1);                }                //开始等待父进程的指令                waitTask(rfd); /*子进程等待任务委派*/                close(rfd);                exit(0);            }            //父进程            int wfd = open(fifoName.c_str(),O_WRONLY);            if(wfd == -1)            {                perror("open");                exit(1);            }            subProcess_.push_back({id,wfd});/*存入子进程信息*/            fifoName.push_back(wfd);        }                   }    void waitTask(int rfd)/*子进程等待任务委派*/    {        while(true)        {            char buff[SIZE];            int n = read(rfd,buff,SIZE);            if(n == -1)            {                perror("read");                exit(1);            }            else if(n>0)            {                Task().curTask((std::string)buff);            }            else if(n == 0)            {                /*证明写端没有写消息了*/                std::cout<<"写端已经关闭,读端也即将关闭!"<>input;               }while(input<0||input>=subProcessNum_);            Task().showTask();            std::string taskName;            std::cout<<"请选择任务#";            std::cin>>taskName;            if(taskName == "exit")            {                break;            }            std::cout<<"选择进程-> "<运行结果
【Linux】命名管道的妙用:实现进程控制与实时字符交互运行结果
3. 实时读取字符

我们还可以通过命名管道来实现字符的实时读取,还挺有意思的,为了实现这个功能,我们不仅需要会使用命名管道,还有如

system()和fflush()
函数。

3.1 公共区域代码语言:javascript代码运行次数:0运行复制
#include #include #include #include #include #include #include #include #define SIZE 1024//命名管道const char* namePipe = "./fifo";//权限掩码const mode_t mode = 0666;
3.2 客户端

客户端用来向服务端发送消息

代码语言:javascript代码运行次数:0运行复制
/** * 该文件为客户端,用来实时向服务端发送字符 */#include "common.hpp"int main(){    /**主要步骤:     * 1.创建命名管道     * 2.打开命名管道     * 3.写入字符到命名管道当中     * 4.关闭命名管道        * */    int mk = mkfifo(namePipe,mode);    if(mk == -1)    {        perror("mkfifo");        exit(1);    }    int fd = open(namePipe,O_WRONLY);//写方式打开    if(fd == -1)    {        perror("open");        exit(1);    }    std::cout<<"开始输入字符:ctrl+c退出"<=0);        (void)n;/*消除未使用警告*/    }    close(fd);    unlink(namePipe);    return 0;}
3.3 服务端代码语言:javascript代码运行次数:0运行复制
/** * 该文件为服务端,用来实时接受客户端输出的字符 */#include "common.hpp"int main(){    /**     * 实时读取客户端发来的字符     * 主要功能:     *      * 1.已读方式打开命名管道文件     * 2.利用fflush实时刷新缓冲区的字符     * 3.关闭文件描述符     */    int fd = open(namePipe,O_RDONLY);    assert(fd!=-1);    while(true)    {        char buff[SIZE]{0};        ssize_t n = read(fd,buff,SIZE-1);        if(n>0)        {            buff[n] = 0;            printf("%c",buff[0]);            fflush(stdout);        }        else if(n == 0)        {            std::cout<效果图

屏幕录制 2024-11-20 202850

往期Linux文章:Linux专栏

4.总结

通过命名管道实现了这两个简单的小程序,其实这些小程序的本质都是一样的:创建命名管道 -> 打开命名管道 -> 通信 -> 关闭命名管道,掌握其中一个即可融会贯通! 感谢阅读。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

359

2023.06.29

如何删除数据库
如何删除数据库

删除数据库是指在MySQL中完全移除一个数据库及其所包含的所有数据和结构,作用包括:1、释放存储空间;2、确保数据的安全性;3、提高数据库的整体性能,加速查询和操作的执行速度。尽管删除数据库具有一些好处,但在执行任何删除操作之前,务必谨慎操作,并备份重要的数据。删除数据库将永久性地删除所有相关数据和结构,无法回滚。

2082

2023.08.14

vb怎么连接数据库
vb怎么连接数据库

在VB中,连接数据库通常使用ADO(ActiveX 数据对象)或 DAO(Data Access Objects)这两个技术来实现:1、引入ADO库;2、创建ADO连接对象;3、配置连接字符串;4、打开连接;5、执行SQL语句;6、处理查询结果;7、关闭连接即可。

349

2023.08.31

MySQL恢复数据库
MySQL恢复数据库

MySQL恢复数据库的方法有使用物理备份恢复、使用逻辑备份恢复、使用二进制日志恢复和使用数据库复制进行恢复等。本专题为大家提供MySQL数据库相关的文章、下载、课程内容,供大家免费下载体验。

256

2023.09.05

vb中怎么连接access数据库
vb中怎么连接access数据库

vb中连接access数据库的步骤包括引用必要的命名空间、创建连接字符串、创建连接对象、打开连接、执行SQL语句和关闭连接。本专题为大家提供连接access数据库相关的文章、下载、课程内容,供大家免费下载体验。

326

2023.10.09

数据库对象名无效怎么解决
数据库对象名无效怎么解决

数据库对象名无效解决办法:1、检查使用的对象名是否正确,确保没有拼写错误;2、检查数据库中是否已存在具有相同名称的对象,如果是,请更改对象名为一个不同的名称,然后重新创建;3、确保在连接数据库时使用了正确的用户名、密码和数据库名称;4、尝试重启数据库服务,然后再次尝试创建或使用对象;5、尝试更新驱动程序,然后再次尝试创建或使用对象。

412

2023.10.16

vb连接access数据库的方法
vb连接access数据库的方法

vb连接access数据库方法:1、使用ADO连接,首先导入System.Data.OleDb模块,然后定义一个连接字符串,接着创建一个OleDbConnection对象并使用Open() 方法打开连接;2、使用DAO连接,首先导入 Microsoft.Jet.OLEDB模块,然后定义一个连接字符串,接着创建一个JetConnection对象并使用Open()方法打开连接即可。

411

2023.10.16

vb连接数据库的方法
vb连接数据库的方法

vb连接数据库的方法有使用ADO对象库、使用OLEDB数据提供程序、使用ODBC数据源等。详细介绍:1、使用ADO对象库方法,ADO是一种用于访问数据库的COM组件,可以通过ADO连接数据库并执行SQL语句。可以使用ADODB.Connection对象来建立与数据库的连接,然后使用ADODB.Recordset对象来执行查询和操作数据;2、使用OLEDB数据提供程序方法等等。

223

2023.10.19

C++ 设计模式与软件架构
C++ 设计模式与软件架构

本专题深入讲解 C++ 中的常见设计模式与架构优化,包括单例模式、工厂模式、观察者模式、策略模式、命令模式等,结合实际案例展示如何在 C++ 项目中应用这些模式提升代码可维护性与扩展性。通过案例分析,帮助开发者掌握 如何运用设计模式构建高质量的软件架构,提升系统的灵活性与可扩展性。

14

2026.01.30

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
麻省理工大佬Python课程
麻省理工大佬Python课程

共34课时 | 5.2万人学习

【web前端】Node.js快速入门
【web前端】Node.js快速入门

共16课时 | 2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号