zoukankan      html  css  js  c++  java
  • Linux学习笔记(13)-进程通信|命名管道

      匿名管道只能在具有亲属关系的进程间通信,那么如果想要在不具有亲戚关系,想在陌生人之间通信,那又该怎么办呢?

      别慌,Linux身为世界上*强大的操作系统,当然提供了这种机制,那便是命名管道……

      所谓命名管道,那便是拥有名字的管道,同时也被称之为FIFO,谈到FIFO,那么做过单片机开发的同学想必是不陌生的。

      在很多单片机的项目中,都使用过FIFO,FIFO其实是一种队列,先进先出,这样可以保证读出数据和写入数据的一致性。

      使用FIFO文件,便可以在不同的,且不具有亲属关系的进程中进程通信。

      创建命名管道的函数叫mkfifo(),关联两个头文件,分别是sys/types.hsys/stat.h

      函数mkfifo的原形如下:

      int mkfifo(const char * pathname, mode_t mode);

      其中参数pathname是,创建的文件名,在这里可以理解为管道名,

      参数mode是,指定所创建的文件的操作权限。

      在mkfifo创建管道之后,需要通过命名管道通信的进程,需要打开该管道文件,然后通过read,write等函数,像操作普通文件一样进行通信。

      现在开始编写代码:

      在两个没有亲戚关系的进程间通信,甲进程向乙进程发送信号,乙进程回复。

      下面是主机端的代码!

    #include<stdio.h>
    #include<stdlib.h>
    #include<sys/types.h>
    #include<sys/stat.h>
    #include<fcntl.h>
    #include<unistd.h>
    #include<errno.h>
    int main(int argc, char * argv[])
    {
        const char * fifoname = "fifo_temp";
        int fd;
        int byte;
        int ret;
        int datafd;
        char buffer[] = {"nihao!"};
    
        if (access(fifoname,F_OK) < 0)//这个函数的说明在最后!
        {
            ret =mkfifo(fifoname,0777);
            if (ret < 0)
            {
                printf("主机端命名管道创建失败!
    ");
                exit(0);
            }
        }
        printf("主机端打开FIFO文件……
    ");
        fd = open(fifoname,O_WRONLY);//不能以O_RDWR模式打开FIFO文件进行读写操作
        printf("errno = %d
    ",errno);
        if (fd > 0)
        {
            printf("主机端打开FIFO成功!
    ");
            printf("主机端开始发送数据!
    ");
            ret = write(fd,buffer,sizeof(buffer));
            if (ret < 0)
            {
                printf("主机端写入FIFO文件失败!
    ");
                exit(0);
            }
            printf("主机端文件写入FIFO成功!
    ");
            close(fd);
        }
        else
        {
            printf("主机端打开FIFO文件失败!
    ");
        }
    
        return 0;
    }

     下面是从机端的代码!

    #include<stdio.h>
    #include<stdlib.h>
    #include<sys/types.h>
    #include<sys/stat.h>
    #include<fcntl.h>
    #include<unistd.h>
    
    int main(int argc, char *argv[] )
    {
        const char *fifoname = "fifo_temp";
        int fd,datafd,ret;
        int byte;
    
        char buffer[1024];
    
        fd = open(fifoname, O_RDONLY);
        printf("从机端打开FIFO文件!
    ");
        if (fd > 0)
        {
            printf("从机端打开FIFO文件成功!
    ");
            byte = read(fd,buffer,1024);
            printf("从机端收到文件为:%s.
    ",buffer);
            close(fd);
        }
    
        return 0;
    }

    编译完毕后,结果如下:

    黄色的那个就是管道文件,看它的开头是用P表示的。

    程序执行结果如下:

      稍微解说一下,首先执行主机端,对于以只写方式(O_WRONLY)打开的FIFO文件,如果open调用是阻塞的(即第二个参数为O_WRONLY)。

      open调用将被阻塞,直到有一个进程以只读方式打开同一个FIFO文件为止由于管道的操作open是阻塞的,所以主机端将会在

      fd = open(fifoname,O_WRONLY);
     这条代码中等待,只有在从机端启动后,以只读方式,同样也打开了那条管道,主机端才会继续执行下去。
    如果不想阻塞,那么需要在open函数的调用的第二个参数中,设置选项O_NONBLOCK,选项O_NONBLOCK表示非阻塞,
     加上这个选项后,表示open调用是非阻塞的,如果没有这个选项,则表示open调用是阻塞的。
    ————————————————————————————————————————————————————————————————————————————————————————————————————————————

      这里为什么要加个与符号&呢?是为了让程序在后台执行,这样我们才能在主机端运行过程中,启动从机端的程序。

      当然,你也可以将主机端的程序设为守护进程,这样也能达到同样的效果。

    ————————————————————————————————————————————————————————————————————————————————————————————————————————————
     知识小结:
    在主机端中,我使用了一个名叫access的函数。

     access():判断是否具有存取文件的权限

    表头文件
        #include<unistd.h>
    定义函数
        int access(const char * pathname, int mode);
    函数说明
        access()会检查是否可以读/写某一已存在的文件。参数mode有几种情况组合, R_OK,W_OK,X_OK 和F_OK。R_OK,W_OK与X_OK用来检查文件是否具有读取、写入和执行的权限。

      F_OK则是用来判断该文件是否存在。由于access()只作权限的核查,并不理会文件形态或文件内容,因此,如果一目录表示为“可写入”,表示可以在该目录中建立新文件等操作,而非意味此目录可以被当做文件处理。例如,你会发现DOS的文件都具有“可执行”权限,但用execve()执行时则会失败。
    返回值
        若所有欲查核的权限都通过了检查则返回0值,表示成功,只要有一权限被禁止则返回-1。

  • 相关阅读:
    [UE4]Image
    [UE4]CheckBox
    [UE4]Button
    [UE4]Border
    [UE4]RichTextBlock
    [UE4]Text Box
    [UE4]字体材质
    [UE4]Retainer Box
    [UE4]Invalidation Box
    [UE4]Dynamic Entry Box
  • 原文地址:https://www.cnblogs.com/han-bing/p/6095859.html
Copyright © 2011-2022 走看看