Linux。 C 中的管道。将数据发送到另一个进程。写作和阅读

Posted

技术标签:

【中文标题】Linux。 C 中的管道。将数据发送到另一个进程。写作和阅读【英文标题】:Linux. Pipes in C. Sending data to another process. Writing and reading 【发布时间】:2016-12-16 12:43:35 【问题描述】:

(警告 - 不是完美的英语!)

在拼贴画上,他们给了我一项任务,我为此苦苦挣扎了几个小时,却无法解决。 我在将数据从一个进程发送到另一个进程时遇到问题。在下面的代码中,我将输入数据发送到子进程,但我没有从子进程接收它。各位大神能帮忙解决一下吗?

程序描述:我的工作是从用户那里获取输入,使用管道将其发送到子进程,进行一些计算,然后将这些计算进一步发送到下一个进程。最后一个过程对传入的数据求和并将其分配给最终结果。更具体地说,第一个子进程计算麦克劳林级数的单个项并将其发送到求和的进程。代码如下:

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>
#include <string.h>
#include <math.h>
#include <sys/stat.h>

int mkfifo(const char *s, mode_t m) 
  return 0;


 int i = 1;
 double FINAL_RESULT = 0 ;


double factorial(int N)

    if(N==1)return 1.0;

    return N*factorial(N - 1);


double Maclaurine(int i , double x , double eps)

    double result;

     result = (pow(x,(double)i) / (double)factorial(i));

    if( result < eps) return 0;

    return result;



void print(char * tab, int size)

    printf("\n");   

    for( int i = 0; i < size ; i++)

        printf("%c" , tab[i]);  

       
    printf("\n");   




int main()

    int pd;

    if(mkfifo("tmp/myfifo1", 0666) == -1)

        printf("Failed to create myfifo2\n");
    

    if(mkfifo("tmp/myfifo2", 0666) == -1)

        printf("Failed to create myfifo2\n");

    

    pd = open("tmp/myfifo1", O_WRONLY);

    double x1,eps1;

    printf("X Value : \n");
    scanf("%lf", &x1);
    printf("EPS Value : \n");
    scanf("%lf", &eps1);

    char tab1[128], tab2[128];

    //converting to char*
    sprintf(tab1,"%lf",x1);
    sprintf(tab2,"%lf",eps1);

    //printing to standard output
    print(tab1, strlen(tab1));
    print(tab2, strlen(tab2));

    //creating new process
    if(fork() == 0)

        //creating another process
        if(fork() == 0)

            int pd3; 
            pd3 = open("tmp/myfifo2", O_RDONLY);

            char tab5[128];

            double x3;

            //reading data from pipe , myfifo2
            read(pd3,tab5,sizeof(tab5));

            //converting char* to float
            x3 = atof(tab5);

            FINAL_RESULT += x3;
            printf("\nCurrent result = %lf" , FINAL_RESULT);

        
        int pd2;
        //opening myfifo1 for reading 
        pd2 = open("tmp/myfifo1", O_RDONLY);

        if(pd2 == - 1)

            printf("\nFailed to open myfifo1 for reading!\n");

        
        double x2, eps2;
        char tab3[128], tab4[128];

        //reading inputed parameters        
        read(pd2,tab3, sizeof(tab3) );

        read(pd2,tab4, sizeof(tab4) );
        //converting to floats
        x2 = atof(tab3);

        eps2 = atof(tab4);

        printf("\nReceived initial data x=%f, eps = %f \n" , x2, eps2);

        double singleTerm = 0;

        int pd4;

        char wyraz[128];

        //opening myfifo2 for writing 
        pd4 = open("tmp/myfifo2", O_WRONLY);

        if(pd4 == -1 )
            printf("\nfailed to open myfifo2\n");
        

        do
            //calculate single term of Maclaurin series
            singleTerm = Maclaurine(i,x2,eps2);
            printf("\n%d. Series term = %lf\n", i,singleTerm);

            sprintf(wyraz,"%lf",singleTerm);

            if(write(pd4, wyraz, sizeof(wyraz)) == -1 )
                printf("\nfailed to send singleTerm to myfifo2\n");
            

            i++;
            //if its less than eps, loop ends
        while(singleTerm);

        printf("\nFinal Result :  %lf", FINAL_RESULT);
        close(pd4);
        close(pd2);

        unlink("tmp/myfifo2");

    

    //write data to myfifo1
    write(pd, tab1, sizeof(tab1));

    write(pd, tab2, sizeof(tab2));

    close(pd);

    unlink("tmp/myfifo1");

    return 0;


这里是代码的结果:

pawel@IMPERIUMVB:~$ gcc pipes.c -lm -o x
pawel@IMPERIUMVB:~$ ./x
X Value : 
10
EPS Value : 
0.001

10.000000
0.001000

pawel@IMPERIUMVB:~$ 
Failed to open myfifo1 for reading!
Received initial data x=0.000000, eps = 0.000000 
failed to open myfifo2
1. Series term = 0.000000
failed to send singleTerm to myfifo2
Final Result :  0.000000
Current result = 0.000000
Failed to open myfifo1 for reading!
Received initial data x=0.000000, eps = 0.000000 
failed to open myfifo2
1. Series term = 0.000000
failed to send singleTerm to myfifo2
Final Result :  0.000000

【问题讨论】:

我只想补充一点,我没有完成这项任务,因为这是课堂作业。我发布它是因为我想了解这里发生了什么。 好的,谢谢,我会改正的。 好的,现在更正了。我希望它现在很好看。 在你的程序中包含 int mkfifo(const char *s, mode_t m) return 0; 意味着你不会创建 FIFO,所以你不能使用它们。您需要删除它才能使用系统调用。你应该总是检查open()的结果——仅仅因为你创建了一个文件(你还没有创建)并不意味着当你去打开它时它仍然可以访问或存在。 是的,你是对的。我只是复制粘贴这个功能,甚至没有考虑只是因为我用谷歌搜索它并且它给了我更少的错误。经过多次尝试,现在我有了解决方案。我将在下面发布代码。感谢您的回复:) 【参考方案1】:

好的,我解决了这个问题,程序给了我预期的结果。 第一个关键要素是擦除

int mkfifo(const char *s, mode_t m)  return 0; 

就像 Jonathan Leffler 说的 *“在你的程序中包含 int mkfifo(const char s, mode_t m) return 0; 意味着你不会创建 FIFO,所以你不能使用它们。你需要删除它以使用系统调用。”

第二件事是支持我的 fifo 在 lseek() 函数单次使用后一遍又一遍地开始。此外,我还必须使用 memset() 清除所有缓冲区。

这是我现在的代码:

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>
#include <string.h>
#include <math.h>
#include <sys/stat.h>



 int i = 0;
 double FINAL_RESULT = 1.0 ;


double factorial(int N)

    if(N==1 || N == 0)return 1.0;

    return N*factorial(N - 1);


double Maclaurine(int i , double x , double eps)

    double result;

     result = (pow(x,(double)i) / (double)factorial(i));

    if( result < eps) return -1.0;

    return result;



void print(char * tab, int size)

    printf("\n");   

    for( int i = 0; i < size ; i++)

        printf("%c" , tab[i]);  

       
    printf("\n");   




int main()

    umask(0);

    if(mkfifo("myfifo1", 0666) == -1)

        printf("Failed to create myfifo2\n");
    

    if(mkfifo("myfifo2", 0666) == -1)

        printf("Failed to create myfifo2\n");

    


    if(fork() == 0)

        int pd = open("myfifo1", O_WRONLY);

        double x1,eps1;
        char tab1[256];
        //assigning tab1 data to 0's

        lseek(pd,0,0);
        memset(tab1, 0, 256);

        printf("X Value : \n");
        scanf("%lf", &x1);


        sprintf(tab1,"%lf",x1);
        //write data to myfifo1
        write(pd, tab1, strlen(tab1));

        printf("EPS Value : \n");
        scanf("%lf", &eps1);

        lseek(pd,0,0);
        memset(tab1, 0, 256);

        //converting to char*
        sprintf(tab1,"%lf",eps1);


        write(pd, tab1, strlen(tab1));

        lseek(pd,0,0);
        memset(tab1, 0, 256);

        close(pd);

        unlink("myfifo1");


        //printing to standard output
        //print(tab1, strlen(tab1));
        //print(tab2, strlen(tab2));

        //creating new process
    
    else if(fork() == 0)


        int pd2, pd3;
        //opening myfifo1 for reading 
        pd2 = open("myfifo1", O_RDONLY);
        //opening myfifo2 for writing 
        pd3 = open("myfifo2", O_WRONLY);



        if(pd2 == - 1)
            printf("\nFailed to open myfifo1 for reading!\n");
        


        if(pd3 == -1 )
            printf("\nfailed to open myfifo2\n");
        

        double x2, eps2;
        char tab2[256];

        lseek(pd2,0,0);
        memset(tab2, 0, 256);

        //reading inputed parameters        
        read(pd2,tab2, sizeof(tab2) );

        //converting to floats
        x2 = atof(tab2);

        lseek(pd2,0,0);
        memset(tab2, 0, 256);

        read(pd2,tab2, sizeof(tab2) );
        //converting to floats

        eps2 = atof(tab2);

        lseek(pd2,0,0);
        memset(tab2, 0, 256);

        //printf("\nReceived initial data x=%f, eps = %f \n" , x2, eps2);

        double singleTerm = 0.0;


        char wyraz[256];



        while(1)

            //calculate single term of Maclaurin series

            singleTerm = Maclaurine(i,x2,eps2);

            if(singleTerm > 0.0)printf("\n%d. Term of Mac Series = %lf", i+ 1,singleTerm);

            sprintf(wyraz,"%lf",singleTerm);


            lseek(pd3,0,0);

            if(write(pd3, wyraz, sizeof(wyraz)) == -1 )
                printf("\nfailed to send singleTerm to myfifo2\n");
            
            if(singleTerm < 0.0)

            //printf("\nI just broke the 1 loop\n");
            memset(wyraz, 0, 256);
            break;
            


            i=i+1;
        
        sleep(1);


        close(pd3);
        close(pd2);

        unlink("myfifo2");
        unlink("myfifo1");

    //creating another process
    else if(fork() == 0)

            int pd4; 
            pd4 = open("myfifo2", O_RDONLY);

            char tab3[256];

            double x3 = 0.0;

            while(1)

            x3=0.0;
            lseek(pd4,0,0);
            memset(tab3, 0, 256);
            //reading data from pipe , myfifo2
            read(pd4,tab3,sizeof(tab3));

            //converting char* to float
            x3 = atof(tab3);

            FINAL_RESULT += x3;


            if(x3<=0.0)

            printf("\nFinal Result :  %lf", FINAL_RESULT ); 
            break;
            //else
            //printf("\n x = %lf, Current result = %lf" ,x3, FINAL_RESULT);
            

            sleep(1);

            printf("\nFinal Result :  %lf", FINAL_RESULT);

            close(pd4);
            unlink("myfifo2");

        

    sleep(7);
    unlink("myfifo1");
    unlink("myfifo2");

    while(1)pause();

    return 0;


现在我可以通过使用麦克劳林级数看到 e^x 函数的数学估计之美。以下是结果:

pawel@IMPERIUMVB:~$ gcc rury.c -lm -o zad1
pawel@IMPERIUMVB:~$ ./zad1
X Value : 
8
EPS Value : 
0.0001

1. Term of Mac Series = 1.000000
2. Term of Mac Series = 8.000000
3. Term of Mac Series = 32.000000
4. Term of Mac Series = 85.333333
5. Term of Mac Series = 170.666667
6. Term of Mac Series = 273.066667
7. Term of Mac Series = 364.088889
8. Term of Mac Series = 416.101587
9. Term of Mac Series = 416.101587
10. Term of Mac Series = 369.868078
11. Term of Mac Series = 295.894462
12. Term of Mac Series = 215.195972
13. Term of Mac Series = 143.463982
14. Term of Mac Series = 88.285527
15. Term of Mac Series = 50.448873
16. Term of Mac Series = 26.906065
17. Term of Mac Series = 13.453033
18. Term of Mac Series = 6.330839
19. Term of Mac Series = 2.813706
20. Term of Mac Series = 1.184718
21. Term of Mac Series = 0.473887
22. Term of Mac Series = 0.180529
23. Term of Mac Series = 0.065647
24. Term of Mac Series = 0.022834
25. Term of Mac Series = 0.007611
26. Term of Mac Series = 0.002436
27. Term of Mac Series = 0.000749

Final Result :  2980.957900 // after adding all of above

e^8 ~ 2980.9579870417

估计是令人满意的:)

【讨论】:

以上是关于Linux。 C 中的管道。将数据发送到另一个进程。写作和阅读的主要内容,如果未能解决你的问题,请参考以下文章

利用消息将一个进程的数据发送到另一个进程(Linux)

如何在Linux上的c中的父进程和子进程之间进行乒乓球

C/C++ 进程间通信 管道

Linux--进程通信

Linux--进程通信

6.Linux进程通信