Linux программирование в примерах
2.4. Переменные окружения
Окружение представляет собой набор пар вида '
' для каждой программы. Эти пары называются переменными окружения. Каждое имя состоит от одной до любого числа буквенно-цифровых символов или символов подчеркивания ('<i>имя</i>=<i>значение</i>'), но имя не может начинаться с цифры. (Это правило контролируется оболочкой; С API может помешать в окружение все, что захочет, за счет возможного запутывания последующих программ.)_Переменные окружения часто используются для управления поведением программ. Например, если в окружении существует
, многие программы запрещают расширения или историческое поведение, которые несовместимы со стандартом POSIX.POSIXLY_CORRECTВы можете решить использовать (и должны задокументировать) переменные окружения для управления поведением вашей программы. Например, вы можете вместо аргумента командной строки использовать для опций отладки переменную окружения. Преимуществом использования переменных окружения является то, что пользователи могут установить их в своем загрузочном файле и не беспокоиться больше постоянным набором определенных опций в командной строке.
Конечно, недостатком использования переменных окружения является то, что они могут молча изменять поведение программы. Джим Мейеринг (Jim Meyering), сопроводитель Coreutils, выразил это таким образом:
Они упрощают пользователю настройку программы без изменения способа ее вызова. Это может быть как благословением, так и проклятием. Если вы пишете сценарий, который зависит от значения определенной переменной окружения, а затем этот сценарий использует еще кто-то, у кого нет таких же установок окружения, он легко может потерпеть неудачу (или, что еще хуже, молча выдать неверные результаты).
2.4.1. Функции управления окружением
Несколько функций позволяют получать значения переменных окружения, изменять эти значения или удалять их. Вот соответствующие объявления:
#include <stdlib.h>char *getenv(const char *name);/* ISO С: Получить переменнуюокружения */int setenv(const char *name, /* POSIX: Установить переменную */const char *value, /* окружения */int overwrite);int putenv(char *string); /* XSI: Установить переменнуюокружения, использует строку */void unsetenv(const char *name); /* POSIX: Удалить переменнуюокружения */int clearenv(void); /* Общее: очистить все окружение */Функция
— та, которую вы будете использовать в 99% случаев. Ее аргументом является имя переменной окружения, которую нужно искать, такое, как «getenv()» или «НОМЕ». Если переменная существует,PATHвозвращает указатель на строковое значение. Если нет, возвращаетсяgetenv(). Например:NULLchar *pathval;/* Поиск PATH; если нет, использовать значениепо умолчанию */if ((pathval = getenv("PATH")) == NULL)pathval = "/bin:/usr/bin:/usr/ucb";Иногда переменная окружения существует, но с пустым значением. В этом случае возвращаемое значение не равно
, но первый символ, на которую оно указывает, будет нулевым байтом, который в С является символом конца строки, 'NULL'. Ваш код должен позаботиться проверить, что возвращаемое значение не равно NULL. Если оно не\0, необходимо также проверить, что строка не пустая, если вы хотите для чего-то использовать значение переменной. В любом случае, не используйте возвращенное значение слепо.NULLДля изменения переменной окружения или добавления к окружению еще одной используется
:setenv()if (setenv("PATH", "/bin:/usr/bin:/usr/ucb", 1) != 0) {/* обработать ошибку */}Возможно, что переменная уже существует в окружении. Если третий аргумент равен true (не ноль), новое значение затирает старое. В противном случае, предыдущее значение не меняется. Возвращаемое значение равно -1, если для новой переменной не хватило памяти, и 0 в противном случае.
для сохранения в окружении делает индивидуальные копии как имени переменной, так и нового ее значенияsetenv()Более простой альтернативой
являетсяsetenv(), которая берет одну строку «putenv()» и помещает ее в окружение:<i>имя</i>=<i>значение</i>if (putenv("PATH=/bin:/usr/bin:/usr/ucb") != 0) {/* обработать ошибку */}слепо заменяет любые предшествующие значения для той же переменной. А также, и это, возможно, более важно, строка, переданнаяputenv(), помещается непосредственно в окружение. Это означает, что если ваш код позже изменит эту строку (например, если это был массив, а не строковая константа), окружение также будет изменено. Это, в свою очередь, означает, что вам не следует использовать в качестве параметров дляputenv()локальную переменную. По всем этим причинамputenv()является более предпочтительной функцией.setenv()ЗАМЕЧАНИЕ. GNU
имеет дополнительную (документированную) особенность в своем поведении. Если строка аргумента является именем без следующего за ним символаputenv(), именованная переменная удаляется. Программа GNU=, которую мы рассмотрим далее в мой главе, полагается на такое поведение.envФункция
удаляет переменную из окружения:unsetenv()unsetenv("PATH");Наконец, функция
полностью очищает окружение:clearenv()if (clearenv() != 0) {/* обработать ошибку */}Эта функция не стандартизирована POSIX, хотя она доступна в GNU/Linux и нескольких коммерческих вариантах Unix. Ее следует использовать, если приложение должно быть очень безопасным и нужно построить собственное окружение с нуля. Если
недоступна, в справке GNU/Linux для clearenv(3) рекомендуется использовать для выполнения этой задачи 'clearenv()'.environ = NULL