структуры. Если он не равен NULL, переменная, на которую он указывает, заполняется значением
val
, a
getopt_long
возвращает 0. Если
flag
не равен NULL, но длинная опция отсутствует, указанная переменная не изменяется.
int val
Если длинная опция обнаружена, это возвращаемое значение или значение для загрузки в
*flag
, если
flag
не равен NULL. Обычно, если
flag
не равен NULL,
val
является
значением true/false, вроде 1 или 0. С другой стороны, если
flag
равен NULL,
val
обычно содержит некоторую символьную константу. Если длинная опция соответствует короткой, эта символьная константа должна быть той же самой, которая появляется в аргументе
optstring
для этой опции. (Все это станет вскоре ясно, когда мы рассмотрим несколько примеров.)
Таблица 2.1. Значения для
has_arg
Макроподстановка
Числовое значение
Смысл
no_argument
0
Опция не принимает аргумент
required_argument
1
Опции требуется аргумент
optional_argument
2
Аргумент опции является необязательным
У каждой длинной опции есть один такой элемент с соответствующими заполненными значениями. В последнем элементе массива все значения должны быть равны нулю. Нет необходимости сортировать массив:
getopt_long
осуществляет линейный поиск. Однако, сортировка его по длинным именам может упростить его чтение для программиста.
При первой встрече использование
flag
и
val
кажется сбивающим с толку. Давайте сделаем на время шаг назад и рассмотрим, почему это работает именно таким способом В большинстве случаев, обработка опций заключается в установке значений различных флаговых переменных при обнаружении различных символов опций, наподобие этого:
while ((с = getopt(argc, argv, ":af:hv")) != -1) {
switch (с) {
case 'a':
do_all = 1;
break;
case 'f':
myfile = optarg;
break;
case 'h':
do_help = 1;
break;
case 'v':
do_verbose = 1;
break;
... /* Здесь обработка ошибок */
}
}
Когда
flag
не равен NULL,
getopt_long
устанавливает значения переменных за вас. Это снижает число операторов
case
в предыдущем
switch
с трех до одного. Вот пример таблицы длинных опций и код для работы с ней:
int do_all, do_help, do_verbose; /* флаговые переменные */
Обратите внимание, что значение, переданное аргументу
optstring
, не содержит больше '
a
', '
h
' или '
v
'. Это означает, что соответствующие короткие опции неприемлемы. Чтобы разрешить как длинные, так и короткие опции, вам придется восстановить в
switch
соответствующие
case
из первого примера.
На практике следует писать свои программы так, чтобы у каждой короткой опции была также соответствующая длинная опция. В этом случае проще всего установить в
flag
NULL, а в
val
соответствующий единичный символ.
2.3.3.2. Длинные опции в стиле POSIX
Стандарт POSIX резервирует опцию
– W
для специфических для производителя возможностей. Поэтому по определению
– W
непереносимо между различными системами.
Если за
W
в аргументе
optstring
следует точка с запятой (обратите внимание не двоеточие),
getopt_long
рассматривает
– Wlongopt
так же, как
– -longopt
. Соответственно в предыдущем примере измените вызов следующим образом: