Linux программирование в примерах
return copy; /* при ошибке возвращает NULL */}С появлением стандарта POSIX 2001 программисты по всему миру могут вздохнуть свободнее: эта функция является теперь частью POSIX в виде расширения XSI:
#include <string.h> /* XSI */char *strdup(const char *str); /* Копировать str */Возвращаемое значение равно
, если была ошибка, или указатель на динамически выделенную память с копиейNULL. Возвращенное значение должно быть освобождено с помощьюstr, когда больше не требуется.free()3.2.3. Системные вызовы:
иbrk()sbrk()Четыре функции, которые мы рассмотрели (
,malloc(),calloc()иrealloc()) являются стандартными, переносимыми функциями для управления динамической памятью.free()На Unix-системах стандартные функции реализованы поверх двух дополнительных, очень примитивных процедур, которые непосредственно изменяют размер адресного пространства процесса. Мы представляем их здесь, чтобы помочь вам понять, как работают GNU/Linux и Unix (снова «под капотом»); крайне маловероятно, что вам когда-нибудь понадобится использовать эти функции в обычных программах. Они определены следующим образом:
#include <unistd.h> /* Обычный */#include <malloc.h> /* Необходим для систем GLIBC 2 */int brk(void *end_data_segment);void *sbrk(ptrdiff_t increment);Системный вызов
действительно изменяет адресное пространство процесса. Адрес является указателем, представляющим окончание сегмента данных (на самом деле, области кучи, как было показано ранее на рис. 3.1). Ее аргумент является абсолютным логическим адресом, представляющим новое окончание адресного пространства. В случае успеха функция возвращает 0, а в случае неуспеха (-1).brk()Функцию
использовать проще; ее аргумент является числом байтов, на которое нужно увеличить адресное пространство. Вызвав ее с приращением 0, можно определить, где в настоящее время заканчивается адресное пространство. Таким образом, чтобы увеличить адресное пространство на 32 байта, используется код следующего вида:sbrk()char *p = (char*)sbrk(0); /* получить текущий конец адресногопространства */if (brk(p + 32) < 0) {/* обработать ошибку */}/* в противном случае, изменение сработало */Практически, вам не нужно непосредственно использовать
. Вместо этого используется исключительноbrk()для увеличения (или даже сокращения) адресного пространства. (Вскоре мы покажем, как это делать, в разделе 3.2.5. «Исследование адресного пространства».)sbrk()Еще более практично вообще никогда не использовать эти процедуры. Программа, которая их использует, не может затем использовать также и
, и это создает большую проблему, поскольку многие элементы стандартной библиотеки полагаются на использованиеmalloc(). Поэтому использованиеmalloc()илиbrk()может приводить к трудно обнаруживаемым крушениям программы.sbrk()Но знать о низкоуровневых механизмах стоит, и конечно же, набор функций
реализован с помощьюmalloc()иsbrk().brk()3.2.4. Вызовы ленивых программистов:
alloca()«Опасность, Билл Робинсон! Опасность!»
Есть еще одна дополнительная функция выделения памяти, о которой вам нужно знать. Мы обсуждаем ее лишь для того, чтобы вы поняли ее, когда увидите, но не следует использовать ее в новых программах! Эта функция называется
; она объявлена следующим образом:alloca()/* Заголовок в GNU/Linux, возможно, не на всех Unix-системах */#include <alloca.h> /* Обычный */void *alloca(size_t size);Функция
выделяетalloca()байтов из стека. Хорошо, что выделенная память исчезает после возвращения из функции. Нет необходимости явным образом освобождать память, поскольку это осуществляется автоматически, как в случае с локальными переменными.sizeНа первый взгляд,
выглядит чем-то типа панацеи для программистов, можно выделять память, о которой можно вовсе не беспокоиться. Подобно Темной Стороне Силы, это, конечно, привлекает. И подобным же образом этого нужно избегать по следующим причинам:alloca()• Функция не является стандартной; она не включена ни в какой стандарт, ни в ISO, ни в С или POSIX.
• Функция не переносима. Хотя она существует на многих системах Unix и GNU/Linux, она не существует на не-Unix системах. Это проблема, поскольку код часто должен быть многоплатформенным, выходя за пределы просто Linux и Unix.
• На некоторых системах
невозможно даже реализовать. Весь мир не является ни процессором Intel x86, ни GCC.alloca()• Цитируя справку [45] (добавлено выделение): «Функция
зависит от машины и от компилятора. На многих системах ее реализация ошибочна. Ее использование не рекомендуется».alloca• Снова цитируя справку: «На многих системах
не может быть использована внутри списка аргументов вызова функции, поскольку резервируемая в стеке при помощиallocaпамять оказалась бы в середине стека в пространстве для аргументов функции».alloca• Она потворствует неряшливому программированию. Тщательная и корректная работа с памятью не сложна; вам просто нужно подумать о том, что вы делаете, и планировать заранее.
GCC обычно использует встроенную версию функции, которая действует с использованием внутритекстового (inline) кода. В результате есть другие последствия
. Снова цитируя справку:alloca()Факт, что код является внутритекстовым (inline), означает, что невозможно получить адрес этой функции или изменить ее поведение путем компоновки с другой библиотекой.
Внутритекстовый код часто состоит из одной инструкции, подгоняющей указатель стека, и не проверяет переполнение стека. Поэтому нет возврата
при ошибке.NULL