Введение в QNX/Neutrino 2. Руководство по программированию приложений реального времени в QNX Realtime Platform
Шрифт:
Функция system — самая простая функция; она получает на вход одну командную строку, такую же, которую вы набрали бы в ответ на подсказку командного интерпретатора, и выполняет ее.
Фактически, для обработки команды, которую вы желаете выполнить, функция system запускает копию командного интерпретатора.
Редактор, который я использую при написании данной книги, использует вызов system. При редактировании мне может понадобиться выйти в командный интерпретатор, проверить некоторые фрагменты программ, и затем снова вернуться в редактор. Все это необходимо сделать, не потеряв позицию курсора в тексте. В этом редакторе я, к примеру, могу дать
Подходит ли функция system для всех дел в Поднебесной? Конечно же, нет. Однако, ее применение может быть очень полезно для множества задач, требующих создания процессов.
Давайте рассмотрим ряд других функций создания процессов.
Следующие функции создания процессов, которые следует рассмотреть, принадлежат к семействам exec и spawn. Прежде, чем мы обратимся к подробностям их применения, рассмотрим суть различий между этими двумя группами функций.
Семейство функций exec подменяет текущий процесс другим. Я подразумеваю под этим то, что когда процесс вызывает функцию семейства exec, этот процесс прекращает выполнение текущей программы и начинает выполнять другую. Идентификатор процесса (PID) при этом не меняется, просто процесс преобразуется в другую программу. Что произойдет с потоками в данном процессе? Мы вернемся к этой теме после того, как рассмотрим функцию fork.
С другой стороны, семейство функций spawn так не делает. Вызов функции семейства spawn создает другой процесс (с новым идентификатором), который соответствует программе, указанной в аргументах функции.
Spawn | POSIX | Exec | POSIX |
---|---|---|---|
spawn | Да | ||
spawnl | Нет | execl | Да |
spawnle | Нет | execle | Да |
spawnlp | Нет | execlp | Да |
spawnlpe | Нет | execlpe | Нет |
spawnp | Да | ||
spawnv | Нет | execv | Да |
spawnve | Нет | execve | Да |
spawnvp | Нет | execvp | Да |
spawnvpe | Нет | execvpe | Нет |
Рассмотрим различные варианты функций exec и spawn. В таблице, представленной ниже, вы увидите, что некоторые функции из них предусмотрены POSIX, а некоторые — нет. Конечно, для максимальной переносимости, следует использовать только POSIX-совместимые функции.
При том, что названия функций могут показаться малопонятными, в их суффиксах есть логика.
Суффикс: | Смысл: |
l (нижний регистр «L») | Список аргументов определяется через список параметров, заданный непосредственно в самом вызове и завершаемый нулевым аргументом NULL. |
е | Указывается окружение. |
p | Если не указано полное имя пути программы, для ее поиска используется переменная окружения PATH. |
v | Список аргументов определяется через указатель на вектор (массив) аргументов. |
Список аргументов здесь — список аргументов командной строки, передаваемых программе.
Заметьте, что в библиотеке языка Си функции spawnlp, spawnvp и spawnlpe все вызывают функцию spawnvpe, которая, в свою очередь, вызывает POSIX-функцию spawnp. Функции spawnle, spawnv и spawnl все в конечном счете вызывают функцию spawnve, которая затем вызывает POSIX-функцию spawn. И, наконец, POSIX-функция spawnp вызывает POSIX-функцию spawn. Таким образом, в основе всех возможностей семейства spawn лежит сам вызов spawn.
Рассмотрим теперь различные варианты функций spawn и exec более подробно так, чтобы вы смогли получить навык свободного использования различных суффиксов. Затем мы перейдем непосредственно к рассмотрению вызова функции spawn.
Например, если я хочу вызвать команду
Или, вариант с применением суффикса v:
Почему именно такой выбор? Он дан для удобства восприятия. У вас может быть синтаксический анализатор, уже встроенный в вашу программу, и может быть удобно сразу оперировать массивами строк. В этом случае я бы рекомендовал применять варианты с суффиксом «v». Или вам может понадобиться запрограммировать вызов программы, когда вам известно, где он находится и какие имеет параметры. В этом случае, зачем вам утруждать себя созданием массива строк, когда вы знаете точно, какие нужны аргументы? Просто передайте их варианту функции с суффиксом «l».