Getopt - Getopt

Getopt это C библиотека функция используется для анализа параметры командной строки стиля Unix / POSIX. Это часть POSIX спецификации, и универсален для Unix-подобный системы.

Это также имя программы Unix для анализа аргументов командной строки в сценариях оболочки.

История

Давняя проблема с программы командной строки было как указать параметры; в ранних программах для этого использовалось множество способов, в том числе односимвольные параметры (), несколько параметров указаны вместе (-abc эквивалентно -a -b -c), многозначные параметры (-inum), варианты с аргументами (-a arg, -inum 3, -a = arg) и разные символы префикса (, + b, / c).

В Getopt функция был написан как стандартный механизм, который все программы могли использовать для анализа параметры командной строки так, чтобы был общий интерфейс, на который можно было положиться. Таким образом, оригинальные авторы выбрали из вариантов поддержки односимвольных параметров, нескольких параметров, указанных вместе, и параметров с аргументами (-a arg или же -aarg), все управляется строкой параметров.

Getopt восходит как минимум к 1980 году[1] и был впервые опубликован AT&T на конференции UNIFORUM 1985 г. в Далласе, штат Техас, с намерением сделать его общедоступным.[нужна цитата ] Его версии впоследствии были перехвачены другими разновидностями Unix (4.3BSD, Linux, так далее.). Это указано в POSIX.2 стандарт как часть unistd.h заголовочный файл. Производные Getopt были созданы для многих языки программирования для анализа параметров командной строки.

Расширения

Getopt является системно-зависимой функцией, и ее поведение зависит от реализации в библиотеке C. Некоторые пользовательские реализации, такие как гнулиб доступны, однако.[2]

Традиционная обработка (POSIX и BSD) заключается в том, что параметры заканчиваются, когда встречается первый аргумент, не являющийся параметром, и что Getopt вернет -1, чтобы сигнализировать об этом. в glibc расширение, однако, варианты разрешены куда угодно для удобства использования; Getopt неявно переставляет вектор аргументов так, чтобы в конце оставались без опций. Поскольку в POSIX уже есть соглашение о возврате -1 при -- и пропуская его, всегда можно переносимо использовать его как указатель конца опций.[2]

А GNU расширение, getopt_long, позволяет анализировать более удобочитаемые параметры с несколькими символами, которые обозначаются двумя дефисами вместо одного. Выбор двух тире позволяет использовать несколько символов (--inum), чтобы отличаться от односимвольных параметров, указанных вместе (-abc). Расширение GNU также допускает альтернативный формат для параметров с аргументами: --name = arg.[2] Этот интерфейс оказался популярным и был принят (без разрешения) многими дистрибутивами BSD, включая FreeBSD а также Солярис.[3] Альтернативный способ поддержки длинных опций можно найти в Solaris и Korn Shell (расширение optstring), но не пользовался такой популярностью.[4]

Другое распространенное расширенное расширение getopt - сброс состояния синтаксического анализа аргументов; это полезно в качестве замены расширения GNU options-anyware или как способ "наслоить" набор интерфейсов командной строки с различными параметрами на разных уровнях. В системах BSD это достигается с помощью optreset переменной, а в системах GNU, установив optind до 0.[2]

Обычная функция-компаньон для Getopt является Getubopt. Он анализирует строку подпараметров, разделенных запятыми.[5]

использование

Для пользователей

Синтаксис командной строки для программ на основе getopt - это рекомендованный POSIX синтаксис аргумента служебной программы. Короче:[6]

  • Опции представляют собой односимвольные буквенно-цифровые символы, которым предшествует - (дефис-минус) символ.
  • Параметры могут принимать аргумент, обязательный или необязательный, или никакой.
  • Когда опция принимает аргумент, он может быть в том же токене или в следующем. Другими словами, если о принимает аргумент, -офу такой же как -o foo.
  • Несколько вариантов могут быть объединены в цепочку, если не последние не принимают аргументов. Если а и б не принимай аргументов пока е принимает необязательный аргумент, -abe такой же как -a -b -e, но -bea не то же самое, что -b -e a из-за предыдущего правила.
  • Все параметры предшествуют аргументам, не являющимся параметром (кроме расширения GNU). -- всегда отмечает конец опций.

Расширения синтаксиса включают соглашение GNU и Sun ЗАЖИМ Технические характеристики.[7][8]

Программистам

В руководстве по getopt от GNU указано такое использование для getopt:[9]

#включают <unistd.h>int Getopt(int argc, char * const argv[],           const char *optstring);

Здесь argc и argv определены точно так же, как в C главный прототип функции, то есть argc указывает длину массива строк argv. В optstring содержит спецификацию того, какие параметры следует искать (обычные буквы и цифры, кроме W) и какие опции принимать аргументы (двоеточия). Например, "vf :: o:" относится к трем вариантам: без аргументов v, необязательный аргумент ж, и обязательный аргумент о. GNU здесь реализует W расширение для длинных синонимов вариантов.[9]

Getopt сам возвращает целое число, которое является либо символом опции, либо -1 для конца опции.[9] Идиома состоит в том, чтобы использовать цикл while для просмотра параметров и использовать оператор switch-case для выбора параметров и действий с ними. См. Раздел с примерами в этой статье.

Чтобы передать дополнительную информацию программе, несколько глобальных внешний переменные используются программой для получения информации из Getopt:

внешний char *optarg;внешний int optind, opterr, оптопт;
optarg
Указатель на аргумент текущего параметра, если он есть. Может использоваться, чтобы контролировать, где начать синтаксический анализ (снова).
optind
Где getopt сейчас смотрит в argv.
opterr
Логический переключатель, определяющий, должен ли getopt выводить сообщения об ошибках.
оптопт
Если возникает нераспознанный параметр, значение этого нераспознанного символа.

Расширение GNU getopt_long интерфейс похож, хотя и принадлежит другому заголовочный файл и принимает дополнительную опцию для определения «коротких» имен длинных опций и некоторых дополнительных элементов управления. Если короткое имя не определено, getopt поместит индекс, ссылающийся на структуру опций в longindex указатель вместо этого.[9]

#включают <getopt.h>int getopt_long(int argc, char * const argv[],           const char *optstring,           const структура вариант *Longopts, int *longindex);

Примеры

Использование стандарта POSIX Getopt

#включают  / * для printf * /#включают  / * для выхода * /#включают  / * для getopt * /int главный (int argc, char **argv) {    int c;    int digit_optind = 0;    int aopt = 0, бопт = 0;    char *копт = 0, *допт = 0;    пока ((c = Getopt(argc, argv, "abc: d: 012")) != -1) {        int this_option_optind = optind ? optind : 1;        выключатель (c) {        дело '0':        дело '1':        дело '2':            если (digit_optind != 0 && digit_optind != this_option_optind)                printf ("цифры встречаются в двух разных элементах argv. п");            digit_optind = this_option_optind;            printf ("вариант% c п", c);            перемена;        дело 'а':            printf ("вариант а п");            aopt = 1;            перемена;        дело 'b':            printf ("вариант б п");            бопт = 1;            перемена;        дело 'c':            printf ("вариант c со значением"% s " п", optarg);            копт = optarg;            перемена;        дело 'd':            printf ("вариант d со значением"% s " п", optarg);            допт = optarg;            перемена;        дело '?':            перемена;        дефолт:            printf ("?? getopt вернул код символа 0% o ?? п", c);        }    }    если (optind < argc) {        printf («неопциональные элементы ARGV:»);        пока (optind < argc)            printf ("% s", argv[optind++]);        printf (" п");    }    выход (0);}

Использование расширения GNU getopt_long

#включают  / * для printf * /#включают  / * для выхода * /#включают  / * для getopt_long; Стандартный getopt POSIX находится в unistd.h * /int главный (int argc, char **argv) {    int c;    int digit_optind = 0;    int aopt = 0, бопт = 0;    char *копт = 0, *допт = 0;    статический структура вариант long_options[] = {    / * ИМЯ АРГУМЕНТ ФЛАГ SHORTNAME * /        {"Добавить",     required_argument, НОЛЬ, 0},        {"добавить",  no_argument,       НОЛЬ, 0},        {"Удалить",  required_argument, НОЛЬ, 0},        {"подробный", no_argument,       НОЛЬ, 0},        {"Создайте",  required_argument, НОЛЬ, 'c'},        {"файл",    required_argument, НОЛЬ, 0},        {НОЛЬ,      0,                 НОЛЬ, 0}    };    int option_index = 0;    пока ((c = getopt_long(argc, argv, "abc: d: 012",                 long_options, &option_index)) != -1) {        int this_option_optind = optind ? optind : 1;        выключатель (c) {        дело 0:            printf ("опции", long_options[option_index].имя);            если (optarg)                printf ("с аргументом% s", optarg);            printf (" п");            перемена;        дело '0':        дело '1':        дело '2':            если (digit_optind != 0 && digit_optind != this_option_optind)              printf ("цифры встречаются в двух разных элементах argv. п");            digit_optind = this_option_optind;            printf ("вариант% c п", c);            перемена;        дело 'а':            printf ("вариант а п");            aopt = 1;            перемена;        дело 'b':            printf ("вариант б п");            бопт = 1;            перемена;        дело 'c':            printf ("вариант c со значением"% s " п", optarg);            копт = optarg;            перемена;        дело 'd':            printf ("вариант d со значением"% s " п", optarg);            допт = optarg;            перемена;        дело '?':            перемена;        дефолт:            printf ("?? getopt вернул код символа 0% o ?? п", c);        }    }    если (optind < argc) {        printf («неопциональные элементы ARGV:»);        пока (optind < argc)            printf ("% s", argv[optind++]);        printf (" п");    }    выход (0);}

В ракушке

Программисты сценариев оболочки обычно хотят предоставить единообразный способ предоставления параметров. Для достижения этой цели они обращаются к getopts и стремятся перенести его на свой язык.

Первой попыткой портирования была программа Getopt, реализованный Лаборатории Unix System (USL). Эта версия не смогла обработать кавычки и метасимволы оболочки, так как не показывает попыток цитирования. Он унаследован от FreeBSD.[10]

В 1986 году USL решила, что небезопасность использования метасимволов и пробелов больше не приемлема, и они создали встроенный Getopts вместо этого для Unix SVR3 Bourne Shell. Преимущество встраивания команды в оболочку состоит в том, что теперь у нее есть доступ к переменным оболочки, поэтому значения можно безопасно записывать без кавычек. Он использует собственные переменные оболочки для отслеживания положения текущей позиции и позиции аргумента, OPTIND и OPTARG, и возвращает имя параметра в переменной оболочки.

В 1995 г. Getopts был включен в Единая спецификация UNIX версия 1 / X / Открыть Руководство по переносимости Выпуск 4.[11] Теперь, будучи частью стандарта POSIX Shell, getopts широко распространились во многих других оболочках, пытающихся быть POSIX-совместимыми.

Getopt было в основном забыто, пока утилита-Linux вышла расширенная версия, которая исправила все старые проблемы getopt путем экранирования. Он также поддерживает длинные имена опций GNU.[12] С другой стороны, длинные варианты редко реализовывались в Getopts команда в других оболочках, кш93 являясь исключением.

На других языках

Getopt представляет собой краткое описание общей структуры аргументов команд POSIX, и оно широко копируется программистами, стремящимися предоставить аналогичный интерфейс как для себя, так и для пользователя в командной строке.

  • C: системы, не относящиеся к POSIX, не поставляются Getopt в библиотеке C, но гнулиб[2] и MinGW (оба принимают стиль GNU), а также некоторые более минимальные библиотеки могут использоваться для обеспечения функциональности.[13] Также существуют альтернативные интерфейсы:
  • D: The Язык программирования D В стандартной библиотеке есть модуль getopt.
  • Идти: поставляется с флаг упаковка,[14] что позволяет использовать длинные имена флагов. В Getopt упаковка [15] поддерживает обработку, близкую к функции C. Есть еще один Getopt упаковка [16] предоставляя интерфейс намного ближе к оригинальному POSIX.
  • Haskell: поставляется с System.Console.GetOpt, который по сути является портом Haskell библиотеки GNU getopt.[17]
  • Java: в стандартной библиотеке Java нет реализации getopt. Существует несколько модулей с открытым исходным кодом, включая gnu.getopt.Getopt, который портирован из GNU getopt,[18] и Apache Commons CLI.[19]
  • Лисп: имеет много разных диалектов без единой стандартной библиотеки. Существует несколько сторонних реализаций getopt для некоторых диалектов Лиспа. Common Lisp имеет известную стороннюю реализацию.
  • Free Pascal: имеет собственную реализацию в качестве одного из стандартных модулей под названием GetOpts. Поддерживается на всех платформах.
  • Язык программирования Perl: имеет две отдельные производные от getopt в своей стандартной библиотеке: Getopt :: Long[20] и Getopt :: Std.[21]
  • PHP: имеет функцию getopt ().[22]
  • Python: содержит модуль в своем стандартная библиотека на основе C getopt и расширений GNU.[23] Стандартная библиотека Python также содержит другие модули для анализа параметров, которые более удобны в использовании.[24][25]
  • Ruby: имеет реализацию getopt_long в своей стандартной библиотеке GetoptLong. В стандартной библиотеке Ruby также есть модули с более сложным и удобным интерфейсом. Доступна сторонняя реализация исходного интерфейса getopt.
  • .NET Framework: не имеет функции getopt в своей стандартной библиотеке. Доступны сторонние реализации.[26]

Рекомендации

  1. ^ ftp://pdp11.org.ru/pub/unix-archive/PDP-11/Distributions/usdl/SysIII/[постоянная мертвая ссылка ]
  2. ^ а б c d е "getopt". GNU Gnulib. Получено 23 января 2020.
  3. ^ getopt_long (3) – FreeBSD Библиотечные функции Руководство
  4. ^ "getopt (3)". Информационная библиотека Oracle Solaris 11.2.
  5. ^ гетсубопт (3) – FreeBSD Библиотечные функции Руководство
  6. ^ «Соглашения об утилитах». POSIX.1-2018.
  7. ^ «Синтаксис аргумента». Библиотека GNU C. Получено 24 января 2020.
  8. ^ Дэвид-Джон, Берроуз; Ковальский III, Джозеф Э. (22 января 2003 г.). «Спецификация CLIP, версия 1.0, PSARC 1999/645» (PDF).
  9. ^ а б c d getopt (3) – Linux Библиотечные функции Руководство
  10. ^ getopt (1) – FreeBSD Общие команды Руководство
  11. ^ "getopts". Открытая группа (POSIX 2018).
  12. ^ getopt (1) – Linux Пользователя Руководство - Пользовательские команды
  13. ^ "visual studio - getopt.h: Компиляция C-кода Linux в Windows". Переполнение стека.
  14. ^ «Пакетный флаг».
  15. ^ "Пакет getopt".
  16. ^ "Пакет getopt".
  17. ^ "System.Console.GetOpt".
  18. ^ "Класс gnu.getopt.Getopt". Получено 2013-06-24.
  19. ^ "Commons CLI". Apache Commons. Фонд программного обеспечения Apache. 27 февраля 2013 г.. Получено 24 июня, 2013.
  20. ^ "Getopt :: Long - perldoc.perl.org".
  21. ^ "Getopt :: Std - perldoc.perl.org".
  22. ^ "PHP: getopt - Руководство".
  23. ^ «16.5. Getopt - синтаксический анализатор в стиле C для параметров командной строки - документация Python 3.6.0».
  24. ^ «Парсер для параметров командной строки». Получено 2013-04-30. Не рекомендуется, начиная с версии 2.7
  25. ^ «Парсер для параметров командной строки, аргументов и подкоманд». Получено 2013-04-30.
  26. ^ "GNU Getopt .NET".

внешняя ссылка