1

これは私にバナナを駆り立てるCでの宿題の練習です!私はそれをほぼ2日間研究していて、まだ機能していないので、どんな助けでも大歓迎です!!

シナリオは次のとおりです。次のようなカンマ区切りのテキストファイルがあります。

(2005 TS15),2013-Apr-07,15.2,0.0391,8.0,0.0205,20.61,20.61,1890,21,13,APO*,0.479,M
(2010 GM23),2013-Apr-13,3.9,0.0099,0.9,0.0022,13.14,13.12,61600,24.7,7,APO*,0.195,M
(2009 SQ104),2013-Apr-22,27.8,0.0714,27.8,0.0714,8.16,8.15,67300,20.9,19,APO*,0.408,S
(2012 XF55),2013-Apr-22,32.7,0.0841,32.7,0.0840,7.33,7.33,15600,22.5,7,APO*,0.446,M

ファイルには10000のようなレコードがあります(地球近傍天体に関連するデータ)。

最大のオブジェクト10個/最も近いオブジェクト10個などを見つける必要があるため、データを構造体の配列に読み込むことを計画しました。各構造体はファイルの列に基づいています。構造体を作成し、ファイルから入力する構造体のインスタンスを作成しました。

私の問題は最後に一般的な機能にあります...私は思う...

構造体の完全な配列を構造体の空白のインスタンスに初期化しようとしましたが、そこでは何も機能しませんでした。

各行を文字列として読み取り、「、」リミッターに基づいてトークン化しようとしましたが、構造体の各要素にトークンを保存する方法がわかりませんでした(コメントアウトされている最後のセクションです... )。

を使用fread()してファイルを構造体の配列に読み込んでみましたが、そこでも何も起こりませんでした(コメントアウトされている最後のセクションです...)

私のコードは以下の通りです。誰かが私を正しい方向に向けることができれば、私は本当に感謝しています!! (フォーマットが正常であることを願っています。問題を適切に説明していない場合は、お知らせください。そうします!!!)

    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>
    #include <math.h>

    #define RECORDS 10972    /* number of records in the file */

    typedef struct {
        char *objectName;
        char *CADate;
        double NomDistLD;
        double NomDistAU;
        double MinDistLD;
        double MinDistAU;
        int VRel;
        int Vinf;
        int NSig;
        float HVm;
        int ref;
        char *ObjClass;
        double Diameter;
        char *Type;
    } NEO; 

    NEO textFile [RECORDS];
    NEO passing[RECORDS];


    void menu();
    void countFile (FILE*fptr);
    void general (FILE*fptr);


    int main (void)
    {
        char filename[50];
        int choice;

        FILE* fptr;

        printf("Enter name of file to open: \n");
        scanf("%s", &filename);

        fptr = fopen(filename, "r");
        if (fptr != NULL)
            printf ("File %s opened succesfully!\n\n", filename);
        else
            printf ("File %s not found!\n", filename);

        countFile(fptr);

       do
       {
     menu();
     choice= 0;

        while(choice <1 || choice >5)
        {
            printf("\nChoose an option from the above (1-5):\n");
            scanf ("%d", &choice);
        }
            switch(choice)
            {
                case 1:
                    printf("function largest\n");
                    break;
                case 2:
                    printf("function closest\n");
                    break;
                case 3:
                    printf("function largest KE\n");
                    break;
                case 4:
                    general (fptr);
                    break;
                default:
                    break;
            }
    }

    while (choice !=5); 
        printf("\n\nProgram Exiting. Goodbye!\n\n");

fclose (fptr);

return 0;
    }



    void general (FILE*fptr)
    {
int count= 0,  fileEnd= 0, i= 0, test= 0;
char **neoArray;
char hold[200];    /* string to hold tokenised strings */
char temp[10000];   /* string to hold each record as it is read in */

char *pToken;
const char *delim = ",";    /* define delimiter to tokenising each record */

passing->objectName = "";
passing->CADate = "";
passing->NomDistLD = 0.0;
passing->NomDistAU = 0.0;
passing->MinDistLD = 0.0;
passing->MinDistAU = 0.0;
passing->VRel = 0;
passing->Vinf = 0;
passing->NSig = 0;
passing->HVm = 0.0;
passing->ref = 0;
passing->ObjClass = "";
passing->Diameter = 0.0;
passing->Type = "";


for (i=0; i<RECORDS; i++)
{
    textFile[i] = passing;
}
    }


      /*  while (!feof(fptr))
{
    test = fread(&passing, sizeof(NEO), 1, fptr);

    if(test != 0)
    {
        printf("\n%s %s %f\n", passing[0].objectName, passing[0].CADate, passing[0].NomDistLD);
    }
}


/*while(!feof (fptr))
{
    for (i=0; i<RECORDS; i++)
    {

        fscanf(fptr, "%s", temp);

        pToken = strtok(temp, delim);
        {
            strcpy(textFile[i].objectName, pToken);
        }

        pToken = strtok(NULL, delim);
        while (pToken != NULL)
        {
            strcpy(textFile[i].CADate, pToken);
        }

        pToken = strtok(NULL, delim);
        while (pToken != NULL)
        {
            textFile[i].NomDistLD = atoi(pToken);
        }

        printf("\n%s %s %f\n", textFile[i].objectName, textFile[i].CADate, textFile[i].NomDistLD);
    }

    rewind (fptr);
}
    }
4

2 に答える 2

1

あなたの問題はおそらく、文字列をメモリのランダムな領域にコピーすることです. objectNameandCADateポインタを初期化しないためです.

これらのポインターを実際の配列に置き換えた方がよいでしょう。

char objectName[128];
char CADate[128];
于 2013-02-28T15:24:01.233 に答える
1

私は各行を読みます

次に、, が見つかるまで順方向にスキャンします。各 , を \0 に置き換えます。それぞれのカウントを保持し、

カウントを使用して別の sscanf 関数を呼び出し、さまざまなフィールドを現在の構造体にロードします。文字列型のフィールドには、メモリを割り当てるために strdup'ing が必要です。

簡単なテストとして、csv ファイルの短いバージョン (たとえば 4 行) を用意し、アサート機能を使用して、期待値が構造体に取り込まれていることを確認します。

この種の問題は、perl や python などのスクリプト言語を使用して行う方が適切です。

お役に立てれば!

于 2013-02-28T15:28:55.373 に答える