Странный printf не работает после пропуска getchar и getchar

Я схожу с ума от C. Я программирую на JavaScript, Ruby, Python, PHP, Lua и даже на Java... Недавно я попытался запрограммировать простой цикл чтения-оценки-печати на языке C. И у меня очень странное поведение с этим простым, базовым кодом в Windows 10. Этот код компилируется с помощью стандартной цепочки инструментов Visual Studio.

  1. иногда (предугадать невозможно) чтение строки пропускается. Я догадался, что в стандартном вводе должен быть какой-то мусор, но я не могу сбросить его с помощью fflush, верно?
  2. каждый раз все printf ПОСЛЕ цикла getchar не работают. Ничего не отображается. Я поставил \n и fflush безрезультатно.

Помоги мне, ты моя единственная надежда.

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

int main(int argc, char * argv[])
{
    const size_t line_length = 1024;
    char * line = malloc(line_length * sizeof(char));
    memset(&line, '\0', line_length);
    char c;
    unsigned short count = 0;
    // fflush(stdin); // tried this, same thing
    printf("Enter: ");
    while ((c = getchar()) != '\n' && c != EOF && count < line_length - 1) {
        line[count] = c;
        count += 1;
    }
    printf("Count : %d\n", count);
    printf("%s\n", line);
    printf("End of program.\n");
    fflush(stdout);
    free(line);
    return EXIT_SUCCESS;
}

Ожидаемый результат:

Enter: hello
Count : 5
hello

Фактический выход (1):

Enter: hello 

Фактический выход (2):

Просто используйте calloc() вместо malloc() + заполните память нулями...

Shawn 16.05.2024 11:51

Вам даже не нужно вызывать memset, вы можете поставить line[count] = '\0'; сразу после цикла while.

Jabberwocky 16.05.2024 14:18
Стоит ли изучать PHP в 2026-2027 годах?
Стоит ли изучать PHP в 2026-2027 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Приемы CSS-макетирования - floats и Flexbox
Приемы CSS-макетирования - floats и Flexbox
Здравствуйте, друзья-студенты! Готовы совершенствовать свои навыки веб-дизайна? Сегодня в нашем путешествии мы рассмотрим приемы CSS-верстки - в...
Тестирование функциональных ngrx-эффектов в Angular 16 с помощью Jest
В системе управления состояниями ngrx, совместимой с Angular 16, появились функциональные эффекты. Это здорово и делает код определенно легче для...
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
3
3
73
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

С вашим кодом есть несколько проблем:

  • ты забыл #include <string.h> ради memset
  • fflush(stdin) имеет неопределенное поведение (хорошо, вы это закомментировали, но это все равно полезно знать)
  • getchar() возвращает int, а не char. Прочтите это для получения дополнительных объяснений.
  • и самое главное: memset(&line, '\0', line_length) неверно, должно быть memset(line, '\0', line_length); (без &), line уже является указателем на память, которую вы хотите установить в 0.

Исправленный код, смотрите мои комментарии, начинающиеся с ///.

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

int main(int argc, char* argv[])
{
  const size_t line_length = 1024;
  char* line = malloc(line_length);
  memset(line, '\0', line_length);      /// removed & 
  int c;                                /// int instead of char
  unsigned short count = 0;
  // fflush(stdin);                     /// fflush(stdin); is UB
  printf(">>> ");
  while ((c = getchar()) != '\n' && c != EOF && count < line_length - 1) {
    line[count] = c;
    count += 1;
  }
  printf("Count : %d\n", count);
  printf("%s\n", line);
  printf("End of program.\n");
  fflush(stdout);
  free(line);
  return EXIT_SUCCESS;
}

@Xitog: важно использовать int c; вместо char c;, потому что char не гарантирует, что сможет представлять значение EOF, тогда как int может. Кроме того, даже если char может представлять значение EOF (что имеет место на большинстве платформ), вы должны использовать int вместо char, потому что в противном случае вы не сможете отличить значение EOF от значения c. реальный персонаж. Однако, как только вы определили, что EOF не имеет значения int, вы можете преобразовать значение char в int c;. Явное приведение типов не требуется.

Andreas Wenzel 16.05.2024 11:38

Незначительное: line_length и count заслуживают того, чтобы быть одного типа. Рекомендую оба size_t.

chux - Reinstate Monica 16.05.2024 16:15

@AndreasWenzel «как только вы определили, что c не имеет значения EOF, вы можете преобразовать значение int в char» --> за исключением того, что педантичное присвоение значения int, такого как UCHAR_MAX, знаковому char вызывает поведение, определяемое реализацией. Очень часто это нормально. Приведение типов подавляет некоторые предупреждения компилятора о сужении line[count] = c;, поэтому приведение типов здесь вполне подходит.

chux - Reinstate Monica 16.05.2024 16:24

Другие вопросы по теме