Хранение данных в списке c - PullRequest
0 голосов
/ 21 февраля 2019

Идея моей программы состоит в том, чтобы считывать данные из файла (в данном случае файл содержит 5 имен) и сохранять их в списке, чтобы позже я мог использовать эти данные, например, для вычисления мин / макс символов.До сих пор я был в состоянии прочитать данные и распечатать их, но вместо того, чтобы печатать, я хотел бы сохранить их в список.Я не мог найти способ как это сделать, поэтому я был бы признателен за помощь.

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

int main(void) {

    char file, filename[25];
    FILE *f;

    printf("Enter the file name: ");
    scanf("%s", filename);

    f = fopen(filename, "r");

    if (f == NULL)
    {
        perror("No file found.\n");
        return 0;
    }

    printf("The contents of %s file are:\n", filename);

    while((file = fgetc(f)) != EOF)
        printf("%c", file);

    fclose(f);

    return 0;
}

Ответы [ 2 ]

0 голосов
/ 22 февраля 2019

Помимо деталей формата ввода, чтение данных не является гибким в вашем текущем решении.Вот более щедрое вводное чтение.Объедините это с другими ответами, и вы должны быть в пути.

#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

int main(void) {
    {
        char *cwd = getcwd(NULL, 0);
        printf("FYI, the current working directory of this program is : `%s'\n", cwd);
        free(cwd);
    }
    printf("Enter the file name: ");
    char *filename;
    int scanf_return = scanf("%m[a-zA-Z./]", &filename);
    if (scanf_return != 1) {
        if (errno != 0) {
            perror("scanf");
        } else {
            fprintf(stderr, "%s\n",
                    "Sorry, unable to read file name. "
                    "Only 'a'...'z', 'A'...'Z', '.' (period) "
                    "and '/' (slash) allowed in the name.");
        }
        return EXIT_FAILURE;
    }
    FILE *f = fopen(filename, "r");
    if (f == NULL) {
        perror(filename);
        free(filename);
        return EXIT_FAILURE;
    }
    printf("The contents of `%s' file are:\n", filename);
    free(filename);
    filename = NULL;

    size_t line_sz = 0u;
    char *line = NULL;
    int nread;
    errno = 0;
    while ((nread = getline(&line, &line_sz, f)) != -1) {
        // If we reached the EOF then there might not be a newline character
        if (line[nread - 1] != '\n') {
            nread++;
        }
        printf("`%.*s'\n", nread - 1, line);
    }
    if (errno != 0) {
        perror("getline");
        free(line);
        fclose(f);
        return EXIT_FAILURE;
    }
    free(line);
    fclose(f);
    return EXIT_SUCCESS;
}
0 голосов
/ 22 февраля 2019

Простой пример использования связанных имен для вашего списка:

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

typedef struct NameList {
  char * name; 
  struct NameList * next;
} NameList;

int append(NameList ** head, NameList ** tail, char * s)
{
  NameList * l;

  if (((l = malloc(sizeof(NameList))) == NULL) ||
      ((l->name = strdup(s)) == NULL))
    /* not enough memory */
    return 0;

  l->next = NULL;
  if (*head == NULL) {
    *head = *tail = l;
  }
  else {
    (*tail)->next = l;
    *tail = l;
  }

  return 1;
}

int main(void) {
  char filename[25];
  FILE * f;

  printf("Enter the file name: ");
  if (scanf("%24s", filename) != 1)
    return 0;

  f = fopen(filename, "r");

  if (f == NULL)
  {
    puts("No file found.");
    return 0;
  }

  NameList * head = NULL;
  NameList * tail = NULL;
  char s[64];

  /* suppose a name has no space even composed and less than 64 characters */
  while (fscanf(f, "%63s", s) == 1) {
    if (!append(&head, &tail, s))
      return 0;
  }

  fclose(f);

  printf("The names in %s file are:\n", filename);

  NameList * l;

  l = head;
  while (l != NULL) {
    puts(l->name);
    l = l->next;
  }

  /* search longer name */
  size_t maxlen = 0;
  char * longer = NULL;

  l = head;
  while (l != NULL) {
    size_t ln = strlen(l->name);

    if (ln > maxlen) {
      maxlen = ln;
      longer = l->name;
    }

    l = l->next;
  }
  if (longer != NULL)
    printf("longer name is : %s\n", longer);

  /* free resources */
  while (head != NULL) {
    l = head;
    head = head->next;
    free(l->name);
    free(l);
  }

  return 0;
}

Компиляция и выполнение

pi@raspberrypi:/tmp $ gcc -pedantic -Wextra l.c
pi@raspberrypi:/tmp $ cat aze
firstname secondname
anothername
lastname
pi@raspberrypi:/tmp $ ./a.out
Enter the file name: aze
The names in aze file are:
firstname
secondname
anothername
lastname
longer name is : anothername

Выполнение в valgrind

pi@raspberrypi:/tmp $ valgrind ./a.out
==10132== Memcheck, a memory error detector
==10132== Copyright (C) 2002-2017, and GNU GPL'd, by Julian Seward et al.
==10132== Using Valgrind-3.13.0 and LibVEX; rerun with -h for copyright info
==10132== Command: ./a.out
==10132== 
Enter the file name: aze
The names in aze file are:
firstname
secondname
anothername
lastname
longer name is : anothername
==10132== 
==10132== HEAP SUMMARY:
==10132==     in use at exit: 0 bytes in 0 blocks
==10132==   total heap usage: 12 allocs, 12 frees, 6,570 bytes allocated
==10132== 
==10132== All heap blocks were freed -- no leaks are possible
==10132== 
==10132== For counts of detected and suppressed errors, rerun with: -v
==10132== ERROR SUMMARY: 0 errors from 0 contexts (suppressed: 6 from 3)

Обратите внимание, что связанный список можно заменить массивом char*, используя realloc , чтобы увеличить его размер при чтении имен и т. Д.

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