Читать книгу 📗 "Linux программирование в примерах - Роббинс Арнольд"
Обратите внимание, что не всегда подходит использование такой оболочки. Если вы сами хотите обработать ошибки, не следует использовать оболочку. С другой стороны, если нехватка памяти всегда является фатальной ошибкой, такая оболочка вполне удобна.
97 if (ferror(ebuf->fp))98 pfatal_with_name(ebuf->floc.filenm);99100 /* Если обнаружено несколько строк, возвратить их число.101 Если не несколько, но _что-то_ нашли, значит, прочитана102 последняя строка файла без завершающего символа конца103 строки; вернуть 1. Если ничего не прочитано, это EOF;104 возвратить -1. */105 return nlines ? nlines : p == ebuf->bufstart ? -1 : 1;106 }В заключение, функция
readline()pfatal_with_name()3.2.1.9. Только GLIBC: чтение целых строк:
getline()getdelim()Теперь, когда вы увидели, как читать строки произвольной длины, вы можете сделать вздох облегчения, что вам не нужно самим писать такую функцию. GLIBC предоставляет вам для этого две функции:
#define _GNU_SOURCE 1 /* GLIBC */#include <stdio.h>#include <sys/types.h> /* для ssize_t */ssize_t getline(char **lineptr, size_t *n, FILE *stream);ssize_t getdelim(char **lineptr, size_t *n, int delim, FILE *stream);Определение константы
_GNU_SOURCEgetline()getdelim()intssize_t<sys/types.h>ssize_tsize_tsize_tОбе функции управляют для вас динамической памятью, гарантируя, что буфер, содержащий входную строку, достаточно большой для размещения всей строки. Их отличие друг от друга в том, что
getline()getdelim()char **lineptrУказатель на
char*getline()NULLmalloc()size_t *nУказатель на размер буфера. Если вы выделяете свой собственный буфер,
*n*nFILE* streamМесто, откуда следует получать входные символы.
По достижении конца файла или при ошибке функция возвращает -1. Строки содержат завершающий символ конца строки или разделитель (если он есть), а также завершающий нулевой байт. Использование
getline()ch03-getline.с/* ch03-getline.c --- демонстрация getline(). */#define _GNU_SOURCE 1#include <stdio.h>#include <sys/types.h>/* main - прочесть строку и отобразить ее, пока не достигнут EOF */int main(void) { char *line = NULL; size_t size = 0; ssize_t ret; while ((ret = getline(&line, &size, stdin)) != -1) printf("(%lu) %s", size, line); return 0;}Вот эта функция в действии, показывающая размер буфера. Третья входная и выходная строки намеренно длинные, чтобы заставить
getline()$ <b>ch03-getline</b> /* Запустить программу */<b>this is a line</b>(120) this is a line<b>And another line.</b>(120) And another line.<b>A llllllllllllllllloooooooooooooooooooooooooooooooonnnnnnnnnnnnnnnngnnnggggggggggg llliiiiiiiiiiiiiiiiiiinnnnnnnnnnnnnnnnnnnneeeeeeeeee</b>(240) A llllllllllllllllloooooooooooooooooooooooooooooooonnnnnnnnnnnnnnnngnnnggggggggggg llliiiiiiiiiiiiiiiiiiinnnnnnnnnnnnnnnnnnnneeeeeeeeee3.2.2. Копирование строк:
strdup()Одной чрезвычайно типичной операцией является выделение памяти для копирования строки. Это настолько типично, что многие программисты предусматривают для нее простую функцию вместо использования внутритекстового кодирования, и часто эта функция называется
strdup()#include <string.h>/* strdup --- выделить память с malloc() и скопировать строку */char *strdup(const char *str) { size_t len; char *copy; len = strlen(str) + 1; /* включить место для завершающего ' ' */ copy = malloc(len); if (copy != NULL) strcpy(copy, str);