Строка AC - это массив char
, оканчивающийся нулевым (нулевым) байтом.Массивы обычно передаются как указатели на первый элемент.Проблема с возвратом этого из функции заключается в том, что указанный адрес должен оставаться действительным в течение срока действия функции, что означает, что он должен быть либо буфером static
(который затем перезаписывается при любых последующих вызовах той же функции,нарушает ранее возвращенные значения) или распределяется функцией, и в этом случае вызывающая сторона отвечает за ее освобождение.
Упомянутый вами scanf
также проблематичен для чтения интерактивного пользовательского ввода, например, он может оставить вводв неожиданном состоянии, например, когда вы не используете символ новой строки в конце строки, вызов next на scanf
(возможно, в несвязанной функции) может неожиданно не дать ожидаемого результата, когда онвстречается с новой строкой.
Часто построчно считывать ввод в буфер, например, с fgets
, а затем анализировать строку оттуда.(Некоторые входные данные вы можете анализировать без буфера, просто читая символ за символом, но такой код часто становится длинным и трудным для быстрого следования.)
Пример чтения любой строки, которая может содержать другие пробелычем символ новой строки, будет выглядеть примерно так:
/// Read a line from stdin and return a `malloc`ed copy of it without
/// the trailing newline. The caller is responsible for `free`ing it.
char *readNewString(void) {
char buffer[1024];
if (!fgets(buffer, sizeof buffer, stdin)) {
return NULL; // read failed, e.g., EOF
}
int len = strlen(buffer);
if (len > 0 && buffer[len - 1] == '\n') {
buffer[--len] = '\0'; // remove the newline
// You may also wish to remove trailing and/or leading whitespace
} else {
// Invalid input
//
// Depending on the context you may wish to e.g.,
// consume input until newline/EOF or abort.
}
char *str = malloc(len + 1);
if (!str) {
return NULL; // out of memory (unlikely)
}
return strcpy(str, buffer); // or use `memcpy` but then be careful with length
}
Другой вариант заключается в том, чтобы вызывающая сторона предоставляла буфер и его размер, а затем просто возвращала тот же буфер в случае успеха и NULL
в случае ошибки.Преимущество этого подхода заключается в том, что вызывающая сторона может решить, когда буфер используется повторно и нужно ли скопировать строку или просто прочитать ее один раз и забыть.