Приложение должно разрешать разворачивание псевдонимов для
popContext
перед первым вызовом функции
poptGetNextOpt
. Псевдонимы для содержимого определяются с помощью трех функций.
int poptReadDefaultConfig(poptContext con, int flags);
Эта
функция считывает псевдонимы из
/etc/popt
и файла
.popt
в домашнем каталоге пользователя. На данный момент
flags
должен иметь нулевое значение, поскольку он зарезервирован только для будущего использования.
int poptReadConfigFile(poptContext con, char * fn);
Файл, определяемый посредством
fn
, открывается и анализируется как конфигурационный файл
popt
. Это позволяет программам использовать конфигурационные файлы конкретных программ.
int poptAddAlias(poptContext con, struct poptAlias alias, int flags);
В некоторых случаях в программах необходимо определять псевдонимы, не читая их из конфигурационного файла. Эта функция добавляет новый псевдоним в содержимое. Аргумент
flags
должен иметь нулевое значение, и в настоящий момент он зарезервирован только для будущего использования. Новый псевдоним определяется как
struct poptAlias
следующим образом:
struct poptAlias {
char * longName; /* может быть NULL */
char shortName; /* может быть '\0' */
int argc;
char ** argv; /*должна быть возможность освобождения с помощью free*/
};
Первые два элемента,
longName
и
shortName
, определяют параметр, для которого вводится псевдоним. Два последних аргумента,
argc
и
argv
, определяют разворачивание, которое будет использовано при обнаружении псевдонима параметра.
26.6. Синтаксический анализ строк аргументов
Хотя библиотека
popt
обычно используется для синтаксического анализа аргументов, уже разделенных на массив вида
argv
, в некоторых программах необходимо анализировать синтаксис строк, формат которых идентичен командным строкам. Для этой цели
popt
предлагает функцию, которая анализирует синтаксис строки в виде массива строки, руководствуясь правилами, подобными обычному синтаксическому анализу в оболочке.
#include <popt.h>
int poptParseArgvString(char * s, int * argcPtr, char *** argvPtr);
Строка
s
разбирается в массив
argv
. Целочисленное значение, на которое указывает второй параметр,
argcPtr
, содержит количество проанализированных элементов, а указатель, на который ссылается последний параметр, указывает на вновь созданный массив. Размещение массива осуществляется динамически; после того как приложение завершит работу с массивом, необходимо вызвать функцию
Некоторые приложения реализуют эквивалент псевдонимов параметров, однако для этого им необходима специальная логика. Функция
poptStuffArgs
позволяет приложению вставлять новые аргументы в текущую структуру
poptContext
.
187
Часто в подобных случаях удобно использовать
POPT_CONTEXT_KEEP_FIRST
.
#include <popt.h>
int poptStuffArgs(poptContext con, char ** argv);
Передаваемый массив
argv
должен иметь указатель
NULL
в качестве своего последнего элемента. Когда функция
poptGetNextContext
будет вызвана в следующий раз, то анализироваться будут сначала "заполненные" аргументы. Библиотека popt возвращает обычные аргументы после того, как закончатся все "заполненные".
26.8. Пример приложения
Библиотека
popt
используется для обработки параметров во многих примерах из других глав книги. Простая реализация
grep
представлена в главе 23, a
robin
— в главе 16. Обе реализации предлагают хорошие примеры использования библиотеки
popt
в большинстве приложений.
RPM, популярная программа для управления пакетами Linux, интенсивно использует функциональные возможности библиотеки
popt
. Многие из ее аргументов командной строки реализованы через псевдонимы
popt
, что делает RPM превосходным примером применения преимуществ
popt
[188] . Более подробную информацию о RPM доступна по адресу http://www.rpm.org.
188
Первоначально библиотека
popt
была реализована для RPM, и многие параметры запросов RPM реализованы в виде простых макросов
popt
.
Программа Logrotate помогает в управлении системными файлами-журналами. Подобно RPM, она являет собой простой пример использования библиотеки
popt
и входит в состав большинства дистрибутивов Linux.
Глава 27
Динамическая загрузка во время выполнения
Загрузка разделяемых (совместно используемых) объектов во время выполнения может оказаться полезным способом для структурирования собственных приложений. Если правильно организовать этот процесс, то тогда можно будет сделать ваше приложение расширяемым, а кроме этого, вы сможете разбивать свой код на логически отдельные модули, что является хорошим стилем написания программ.