0

我有以下代码,我必须输入 2 个参数作为输入,其中:第一个参数是一个数字。如果这个数字是 2 的倍数,那么将使用 fork 创建 15 个子进程。如果数量是 3 的倍数,则将创建 10 个进程。如果数字是 5 的倍数,则将创建 7 个进程。第二个参数是一个文件,它的大小(以字节为单位)将分为创建的进程数,每个子进程将读取文件的一部分,他们将该部分保存到一个变量中,最后父进程显示所有该变量中的文本。例如,我使用 ./p 5 /home/directoryFile.c 运行程序。所以我有 7 个子进程,假设文件大小为 700 字节。这意味着每个子进程都应该读取 100 个字节,他们将其保存到一个变量中(附加内容),最后父级将所有内容一起显示。问题是应该显示父项中所有内容的变量 textToSend 没有显示任何内容……我相信子项中的 sprintf 行应该有问题。

//gcc Test.c -o p
//./p 5 /home/directoryFile.c

#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <dirent.h>
#include <sys/wait.h>

int main(int argc, char *argv[])
{
    int studentId, children = 0, j, i, childNumber[15], fdFile, fdread;
    float bytesToRead;
    char directory[50];
    char *buffer = malloc(256), *textToSend = malloc(256);

    system("clear");

    if(argc-1 < 1)
    {
        printf("\nSome arguments are missing\n");
        return EXIT_FAILURE;
    }

    studentId = atoi(argv[1]);
    strcpy(directory,argv[2]);

    if((studentId%2) == 0)
    {
        children = 15;
    }
    else
    {
        if((studentId%3) == 0)
        {
            children = 10;
        }
        else
        {
            if((studentId%5) == 0)
            {
                children = 7;
            }
            else
            {
                printf("\nStudentId is not multiple of 2, 3 o 5\n");
                return EXIT_FAILURE;
            }
        }
    }

    struct stat fileInfo;
    stat(argv[2],&fileInfo);

    bytesToRead = fileInfo.st_size / children;

    printf("children: %d\n",children);
    printf("File Size: %lld\n",(long long int) fileInfo.st_size);
    printf("Bytes: %.2f\n",bytesToRead);

    fdFile = open(directory,O_RDONLY);

            if(fdFile == -1)
            {
                printf("\nError opening the fileo\n");
                return EXIT_FAILURE;
            }

    for(i=0;i<children;i++)
    {
        childNumber[i] = fork();

        if(childNumber[i] == -1)
        {
            printf("\nError creating the child process\n");
            return EXIT_FAILURE;
        }

        if(childNumber[i] == 0)
        {
            fdread = read(fdFile,buffer,bytesToRead);

            if(fdread == -1)
            {
                printf("\nError reading the file\n");
                return EXIT_FAILURE;
            }

            printf("%s",buffer);
            //printf("\n\n------------------------\n\n");

            sprintf(textToSend,"%s%s",textToSend,buffer);

            return EXIT_SUCCESS;
        }
        else
        {
            //waitpid(childNumber[i],NULL,WNOHANG);
        }
    }

    printf("\nThis is the content of the file: %s\n",textToSend);

    close(fdFile);

    for(j=0;j<children;j++)
    {
        wait(NULL);
    }

    return EXIT_SUCCESS;
}

会发生什么?缓冲区正确显示文件的一部分......它可能是 sprintf 函数吗?

4

1 回答 1

1

您正在使用sprintf打印到您的进程之间不共享的本地缓冲区。

当您fork的孩子收到所有父母记忆的完整私人副本时,这是私人的。默认情况下,所有内存都是私有的,除非专门分配为共享。

如果您将其分配为共享

 char * textToSend = mmap(NULL, 256, PROT_READ | PROT_WRITE, MAP_ANONYMOUS | MAP_SHARED, -1, 0);

然后,您将拥有可从所有进程访问的相同内存。

这不会说明您必须告诉所有其他进程缓冲区的开始已移动,因此需要额外的共享变量和同步访问。目前您没有附加文本sprintf,您正在覆盖任何以前的内容。

最后,如果您希望将来自不同进程的结果汇集在一起​​,您可能希望使用管道 ( pipe(2)) 而不是共享内存。

于 2013-10-07T15:42:35.833 回答