0

こんにちは、マルチプロセス プログラミングの課題があり、文字を 1 行ずつ読み込もうとすると問題が発生します (各行は 3 文字と 1 整数で構成されます)。問題は、私が使用するfscanfと正しく機能せず、2ステップごとに文字を正常に読み取ることです。ここで私のコードを見ることができます。

#include <stdio.h>     /* basic I/O routines.   */
#include <stdlib.h>
#include <unistd.h>    /* define fork(), etc.   */
#include <sys/types.h> /* define pid_t, etc.    */
#include <sys/wait.h>  /* define wait(), etc.   */
#include <signal.h>    /* define signal(), etc. */
#include <pthread.h>
#include <ctype.h>

void child_process(int);
void parent_process();
void function();
int counter=0;
int bond_number=0;

char* f_strand;

int main(int argc, char* argv[]) {

    counter = atoi(argv[1]);

    function();

    fflush(stdout);

    execl("/usr/bin/killall","killall","tail",(char *) 0);
    return 0;
}

void function(){

    int i,k;
    pid_t child_pid[counter];
    char array[counter];
    char array_opst[counter];
    srand ( time(NULL) );
    int temp;

    FILE* fptr;
    fptr = fopen("sample.txt","w");

    if(! (f_strand=(char*)malloc(counter*sizeof(char))) ){
        printf("Error\n");
        exit(1);
    }

    for(i=0; i<counter; i++){

        temp = rand()%4;

        if(temp==0){
            *(f_strand+i) = 'A';
            fprintf(fptr,"A\n");
        }
        else if(temp==1){
            *(f_strand+i) = 'C';
            fprintf(fptr,"C\n");
        }
        else if(temp==2){
            *(f_strand+i) = 'T';
            fprintf(fptr,"T\n");
        }
        else if(temp==3){
            *(f_strand+i) = 'G';
            fprintf(fptr,"G\n");
        }
    }

    fclose(fptr);

                    for (i = 0; i < counter; i++) {
                      if ( (child_pid[i] = fork() ) < 0) {

                        perror("fork");
                        abort();
                      } 
                      else if (child_pid[i] == 0) {

                        child_process(i);
                        exit(0);
                      }
                    }
    int status;
    pid_t pid;
    int num=counter;
        while (num > 0) {
          pid = wait(&status);
          //printf("Child with PID %ld exited with status 0x%x.\n", (long)pid, status);
          --num;  // Remove pid from the pids array.
        }


    parent_process();
    printf("\n");

}

void child_process(int index){

    int i;
    char str,str1,str2;

    FILE* fptr;
    fptr = fopen("sample.txt","r");

    fseek (fptr, 2*index, SEEK_SET);
    fscanf(fptr,"%c", &str);

    FILE* fptr2;
    fptr2 = fopen("sample_2.txt","a");

       if( str == 'A' ){
            str1='T';
            str2=';';
            fprintf(fptr2,"%c%c%c%d\n",str,str2,str1,index);
            }
       else if( str == 'T' ){
            str1='A';
            str2=';';
            fprintf(fptr2,"%c%c%c%d\n",str,str2,str1,index);
            }
       else if( str == 'G' ){
            str1='C';
            str2=':';
            fprintf(fptr2,"%c%c%c%d\n",str,str2,str1,index);
            }
       else if( str == 'C' ){
            str1='G';
            str2=':';
            fprintf(fptr2,"%c%c%c%d\n",str,str2,str1,index);
            }

    fflush(stdout);
    fclose(fptr);
    fclose(fptr2);
    sleep(1);

}

void parent_process(void){

    FILE* fptr;
    fptr = fopen("sample_2.txt","r");

    char str,str1,str2;
    int index,i,k;
    char array_opst[counter]; //  second strand
    char array_type[counter]; // bond type

    i=0;

    while( !feof(fptr) ){

        if(4==fscanf(fptr,"%c%c%c%d",&str,&str1,&str2,&index)){

            if(feof(fptr))
                break;

            printf("STR = %c\n, K = %d",str,k);

            array_type[index]=str1;
            array_opst[index]=str2;
            printf("WHILE\n");

            if(array_type[index] == ':')
                bond_number=bond_number+2;
            else if(array_type[index] == ';')
                bond_number=bond_number+3;
    }

        i++;
}
    fclose(fptr);

}

これは sample2.txt の出力です

C:G5
G:C6
A;T4
C:G3
G:C7
C:G8
G:C2
A;T9
T;A1
C:G0

parent_process()ファイルから3文字と1行ずつ整数を読み取ろうとしますsample2.txt。ただし、このプロセスは 2 行ごとに行われます。つまり、最初の 3 番目の 5 番目の行を読み、続いていきます。とにかく助けてくれてありがとう。

4

2 に答える 2

3

これは興味深い多面的な問題です。最初に注意すべきことは、c変換を使用していることです。C99 標準では次のように規定されています (§7.19.6.2¶8)。

指定に 、、または指定子が含まれていない限り、 (関数で指定された) 入力空白文字isspaceはスキップされます。[cn

それでは、重要な行を見てみましょう。

    if(4==fscanf(fptr,"%c%c%c%d",&str,&str1,&str2,&index)){

何が起こるかというと、空白 (エントリ間のパディング、行末記号など) が によって消費されているということfscanfです。dこれが発生すると、変換が発生したときに整数の読み取りに失敗するなど、読み取りがずれます。その結果、fscanfは に評価されず4、分岐は行われないため、誤って読み取られたデータは出力されません。

代わりに、適切な入力に「再調整」するために 1 行おきに誤って読み取る必要があるため、これらの選択された行のみが出力されます。これを証明するために、条件を単にに置き換えてみてください0 < fscanf(...)。誤った読み取りが出力を生成し、1 行おきの読み取りの中間試行を示します。

于 2012-10-13T17:41:26.500 に答える
2

問題は改行文字だと思います。フォーマット文字列に含めるか、連続する fscanf() 呼び出しの間に追加の文字を読み取ってください。

于 2012-10-13T17:27:45.650 に答える