进程通信1——管道、命名管道

匿名管道
管道是一种简单的进程通信(IPC)机制。管道实质上就是pipe函数在内核中开辟了一段缓冲区,有一个读端和一个写端。两个进程之间能够通信的本质:通过fork函数传递文件描述符(子进程是父进程的副本,父进程所有打开的文件描述符都被复制到子进程中,父子进程的每个相同的打开描述符共享一个文件表项)使得两个进程可以访问同一个管道,从而实现通信。

特点:

  1. 单向通信(若实现双向通信,必须建立两个管道);
  2. 适用于有血缘关系的进程之间,尤其指父子进程;
  3. 面向数据流的服务;
  4. 生命周期随进程;

使用管道(阻塞I/O操作)的特殊情况:

  1. 写端都关闭,读端读取完后返回零;
  2. 写端未关闭但不进行写入,读端读取数据后陷入阻塞,直至有数据写入需要读取;
  3. 读端都关闭,在写端进行写入数据的程序会收到SIGPIPE信号,使得程序异常退出;
  4. 读端未关闭但不进行读取,写端将缓冲区写满后陷入阻塞。

注:设置O_NONBLOCK标志,可设置为非阻塞。

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<string.h>
int main()
{
    int pipefd[2] = {0,0};
    if(pipe(pipefd)<0)
    {
        perror("pipe");
        exit(1);
    }
    pid_t id = fork();
    if(id < 0)
    {
        perror("fork");
        exit(2);
    }
    else if(id == 0)
    {
        //child
        close(pipefd[0]);
        char* const msg = "hello bit";
        int count = 5;
        while(count)
        {
            write(pipefd[1],msg,strlen(msg));
            count--;
        }
        close(pipefd[1]);
        exit(3);
    }
    else
    {
        //father
        close(pipefd[1]);
        int count = 5;
        char buf[1024];
        while(count)
        {
            ssize_t s = read(pipefd[0],buf,sizeof(buf)-1);
            if(s > 0)
            {
                buf[s] = 0;
                printf("%s\n",buf);
            }
            else
            {}

            count--;
        }
        close(pipefd[0]);
        exit(5);
    }
    return 0;
}

命名管道
命名管道是为了解决匿名管道只适用于有血缘关系的进程之间通信的缺陷而设计的。不同于匿名管道的是它提供了一个路径名与其相连,以FIFO的文件形式存在于文件系统中。这样即使两进程不存在血缘关系,但也可以通过访问该路径,实现彼此相互通信。
命名管道的创建

  1. 在shell下交互地建立一个命名管道;
    命令:mknod namedpipe
  2. 使用系统函数mkfifo/mknod;
    int mkfifo(const char* path,mode_t mod,dev_t dev);
    其中path是创建的命名管道的全路径名;mod为创建的命名管道的模式,指明其权限;dev为设备值,该值取决于文件创建的种类。
    mkfifo(“/code/fifo”,S_IFIFO|0666) ; //第二个参数指明创建一个命名管道且存取权限为0666。(注意umask对生成的管道文件权限的影响)
    命名管道使用前,必须先调用open()将其打开,需要注意的是该进程可能被阻塞,但是如果使用读写方式打开,则一定不会导致阻塞。

我们也可以使用下面函数创建命名管道:
mkfifo(const char* filename,mode_t mode); //创建一个真实存在于文件系统中的文件。

两种管道的比较:

  1. 命名管道是一个存在于硬盘上的FIFO文件,而匿名管道是存在于内存的特殊文件;
  2. 命名管道是双向通信,匿名管道是单向通信;
  3. 命名管道可用于网络通信;
  4. 命名管道有两种模式(字节流/字节),匿名管道是面向数据流的;
  5. 两者的生命周期都是随进程(缺陷);
  6. 两者的每条消息的最大长度都是有上限的(缺陷);
//client

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

int main()
{
    umask(0);
    if(mkfifo("./fifo",0666|S_IFIFO) < 0)
    {
        perror("mkfifo");
        exit(1);
    }
    int fd = open("./fifo",O_RDONLY);
    if(fd < 0)
    {
        perror("open");
        exit(2);
    }
    char buf[128];
    while(1)
    {
        ssize_t s = read(fd,buf,sizeof(buf)-1);
        if(s < 0)
        {
            perror("read");
            exit(3);
        }
        else if(s == 0)
        {
            printf("client quie,I should quit\n");
            break;
        }
        else
        {
            buf[s] = 0;
            printf("%s",buf);
        }
    }
    return 0;
}
//server

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

int main()
{
    int fd = open("./fifo",O_WRONLY);
    if(fd < 0)
    {
        perror("open");
        exit(1);
    }

    while(1)
    {
        printf("please enter:");
        fflush(stdout);
        char buf[128];
        ssize_t s = read(0,buf,sizeof(buf)-1);
        if(s < 0)
        {
            perror("read");
            exit(2);
        }
        buf[s] = 0;
        write(fd,buf,sizeof(buf)-1);
    }
    close(fd);
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 Windows 中,可以使用命名管道(Named Pipes)实现进程通信。以下是一些基本步骤: 1. 创建命名管道 使用 CreateNamedPipe 函数创建一个命名管道。该函数需要指定管道名称、管道的读写模式、管道的最大实例数等参数。 2. 等待客户端连接 使用 ConnectNamedPipe 函数等待客户端的连接。该函数会一直阻塞,直到有客户端连接成功。 3. 接收客户端数据 使用 ReadFile 函数从管道中读取客户端发送的数据。 4. 发送数据给客户端 使用 WriteFile 函数向管道中写入数据,以便客户端读取。 5. 断开连接 使用 DisconnectNamedPipe 函数断开与客户端的连接。如果需要与多个客户端通信,则返回第 2 步。 6. 关闭管道 使用 CloseHandle 函数关闭命名管道的句柄。 注意事项: - 在创建管道时,需要指定管道名称,该名称在系统中必须是唯一的。 - 管道支持同步和异步方式进行读写操作,可以根据具体需求选择使用哪种方式。 - 管道的读写操作是阻塞式的,也可以使用 overlapped 结构体实现异步操作。 下面是一个简单的代码示例,演示如何使用命名管道实现进程通信: ``` #include <windows.h> #include <stdio.h> #define PIPE_NAME "\\\\.\\pipe\\MyPipe" int main() { HANDLE hPipe; char buffer[1024]; DWORD dwRead; // 创建命名管道 hPipe = CreateNamedPipe(PIPE_NAME, PIPE_ACCESS_DUPLEX, PIPE_TYPE_BYTE | PIPE_READMODE_BYTE | PIPE_WAIT, PIPE_UNLIMITED_INSTANCES, 0, 0, 0, NULL); if (hPipe == INVALID_HANDLE_VALUE) { printf("CreateNamedPipe failed! Error code: %d\n", GetLastError()); return 1; } // 等待客户端连接 if (!ConnectNamedPipe(hPipe, NULL)) { printf("ConnectNamedPipe failed! Error code: %d\n", GetLastError()); return 1; } // 接收客户端数据 if (!ReadFile(hPipe, buffer, sizeof(buffer), &dwRead, NULL)) { printf("ReadFile failed! Error code: %d\n", GetLastError()); return 1; } printf("Received data from client: %s\n", buffer); // 发送数据给客户端 if (!WriteFile(hPipe, "Hello, client!", 15, NULL, NULL)) { printf("WriteFile failed! Error code: %d\n", GetLastError()); return 1; } // 断开连接 DisconnectNamedPipe(hPipe); // 关闭管道 CloseHandle(hPipe); return 0; } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值