Каков наилучший способ извлечь подстроки из более длинной строки в C - PullRequest
0 голосов
/ 24 мая 2019

У меня есть файл, отформатированный как строка, за которой следует длинный список с плавающей точкой, разделенных пробелами:

строка (пробел) float (пробел) ... float

строка (пробел)float (пробел) ... float

.

.

.

строка (пробел) float (пробел) ... float

Строки и числа каждой строки должны быть помещены в структуру, и в настоящее время я делаю это, сохраняя каждую строку в виде строки, используя fgets, а затем увеличивая ее по этой строке, проверяя подстроки между пробелами, а затемпреобразование этих строк в float и сохранение их в моей структуре.

Это становится очень утомительным и очень сложным.Есть ли лучший способ сделать это?

1 Ответ

1 голос
/ 24 мая 2019

В зависимости от количества переменных, которые должны быть фиксированными или переменными, возможны два подхода.В случае фиксированного числа операций с плавающей запятой возможное решение может быть:

файл 'test_data.txt' с данными:

test1 1.41 1.73 2.78 3.14
test2 2.41 2.73 3.78 4.14

и исходный файл для чтения данных может быть:

#include <stdio.h>

int main(int argc, char ** argv)
{
        FILE * file = fopen("test_data.txt", "r");
        if (file == NULL)
        {
                printf("Cannot open file.\n");
                return 1;
        }

        char string[32] = { 0 };
        float f1, f2, f3, f4;
        while (feof(file) == 0)
        {
                fscanf(file, "%s %f %f %f %f", string, & f1, & f2, & f3, & f4);
                printf("For string: %s values are:\n\t%f %f %f %f\n", string, f1, f2, f3, f4);
        }

        fclose(file);
        return 0;
}

Но, учитывая, что вы указали, что число с плавающей точкой варьируется, возможное решение могло бы быть таким:

файл 'test_data.txt' с данными:

test1 1.41 1.73 2.78 3.14
test2 2.41 2.73 3.78 4.14 5.15

и исходный файл для чтения данных может быть:

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

static void remove_trailing_spaces(char ** begin)
{
        while (isspace(** begin))
                ++(* begin);
        return;
}

static void get_string(char ** begin, char * output)
{
        remove_trailing_spaces(begin);

        char * end = * begin;
        while (isalnum(* end))
                ++end;

        strncpy(output, * begin, (int)(end - * begin));
        * begin = end;
        return;
}

static void get_float(char ** begin, float * output)
{
        remove_trailing_spaces(begin);

        char * end;
        * output = strtof(* begin, & end);
        * begin = end;
        return;
}

int main(int argc, char ** argv)
{
        FILE * file = fopen("test_data.txt", "r");
        if (file == NULL)
        {
                printf("Cannot open file\n");
                return 1;
        }

        char buffer[1024] = { 0 };
        char string[32] = { 0 };
        while (feof(file) == 0)
        {
                if (fgets(buffer, 1024, file) != NULL)
                {
                        char * begin = & buffer[0];
                        get_string(& begin, string);
                        printf("For string: %s values are:\n\t", string);

                        while ((feof(file) == 0) && (* begin != '\n'))
                        {
                                float f = 0.0;
                                get_float(& begin, & f);
                                printf("%f ", f);
                        }
                        printf("\n");
                }
        }
        fclose(file);
        return 0;
}

Имейте в виду, что это может быть не самым лучшим решением.Это только показывает, что синтаксический анализ текстового файла с изменяющимся количеством данных в каждой строке файла test_data.txt немного больше усилий, чем в первом случае.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...