Command line parameters and environment variables/ru
│
English (en) │
español (es) │
suomi (fi) │
français (fr) │
русский (ru) │
В большинстве (интерактивных) операционных систем программы можно запускать через интерфейс командной строки command line interface (CLI), что позволяет предоставлять программе дополнительные данные. Модули system (и objPas
) предоставляют базовые функции для доступа к данным, предоставленным из командной строки.
Обзор
Хотя на этой странице говорится о параметрах командной строки и переменных среды, фактический интерфейс командной строки (CLI) не является обязательным требованием для использования таких данных, а также для использования процедур и терминологии, описанных здесь. Однако, поскольку интерфейс командной строки более осязаем, следующие инструкции в основном относятся к пользовательскому интерфейсу оболочки.
Терминология
- Option (переключатели)
- Опции - флаги yes/no. Обычно они имеют вид
‑‑dryrun
/‑‑no‑dryrun
(dryrun включен или отключен соответственно). Опции также называются «переключателями». - Parameter (параметр)
- Параметры представляют собой кортежи ключ-значение. В командной строке они выглядят так:
‑‑processors 4
. - Argument (аргумент)
- Аргументы — это простые слова. В командной строке они обычно разделяются пробелами (назначают переменную IFS в некоторых оболочках). В узком смысле, аргументы — это все предоставленные слова, которые не являются ни опциями, ни (частью) параметров. В общем, аргументы — это все слова в командной строке, которые никак не интерпретируются.
- Environment variable (переменные окружения)
- Переменные окружения относятся к помеченной части хранилища в среде. Это пары имя-значение.
Обратите внимание, CLI/OS предоставляет программе только те аргументы, которые она должна получить: например, экранированные символы новой строки, перенаправление файла (каналы или в файл) и присвоение переменных среды не будут перенаправлены в программу.
Основы
Поскольку понятия «опция» и «параметры» уже являются своего рода высокоуровневыми, существуют разные стили (короткие опции против длинных опций), и модуль system стремится предоставить базовые средства для работы, все аргументы командной строки просто перечисляютсяить, начиная с нуля. Никакой интерпретации не делается.
Функция paramStr
возвращает n-й аргумент в командной строке. ParamStr(0)
пытается вернуть имя и, возможно, полный путь к исполняемому программному файлу.
Параметры командной строки
Основы
Программа на Паскале может получить доступ к аргументам командной строки, используя функциюparamStr
в сочетании с paramCount
. ParamCount
возвращает количество предоставленных аргументов.
program listArguments(input, output, stdErr);
{$mode objFPC}
var
i: integer;
begin
writeLn({$ifDef Darwin}
// в Mac OS X возвращаемое значение зависит от метода вызова
'This program was invoked via: ',
{$else}
// Turbo Pascal-совместимый paramStr(0) возвращает местоположение
'This program is/was stored at: ',
{$endIf}
paramStr(0));
for i := 1 to paramCount() do
begin
writeLn(i:2, '. argument: ', paramStr(i));
end;
end.
Запуск этой программы (на не-Mac OS Ⅹ платформе) может выглядеть следующим образом:
$ ./listArguments foo bar able
This program is/was stored at: /tmp/listArguments
1. argument: foo
2. argument: bar
3. argument: able
Модуль RTL system
предоставляет версию paramStr
, которая возвращает короткие строки shortString
. Из-за особенностей реализации shortString
ограничены 255 символами. Однако пользователь может указать более длинные аргументы командной строки.
Чтобы получить к ним доступ, модуль objPas
переопределяет paramStr
, который вместо этого возвращает строки ansiString
, которые не имеют этого ограничения длины. Модуль objPas
автоматически включается в режимы компилятора {$mode objFPC}
(строка2) и {$mode Delphi}
.
Распределенная функция paramStr
пытается быть совместимой с Turbo Pascal. В TP paramStr(0)
возвращает местоположение программы. Однако операционная система должна поддерживать это. В частности, в Mac OS Ⅹ значение paramStr(0)
зависит от метода вызова, от того, как запускается приложение.
paramStr(0)
зависит от функциональности ОС, он не подходит для кроссплатформенных программ.paramStr(0)
на юниксоидных платформах бесполезен.Модуль getOpts
предоставляет несколько других подпрограмм для вызова программы в соответствии с определенным стилем.
Простота использования
Хорошая программа должна выдавать справочное сообщение при вызове с неправильными параметрами и должна следовать общепринятому способу предоставления параметров. Модуль custApp
, поставляемый с FPC, предоставляет класс TCustomApplication
, предоставляющий функции для простой проверки и чтения параметров. Конечно, вы по-прежнему можете получить доступ к параметрам напрямую через paramStr
и paramCount
.
Каждое приложение LCL использует это автоматически. Объект Application — это TCustomApplication.
Если вы хотите написать не LCL-программу, то создайте в Lazarus новый проект типа «Консольное приложение». Это создаст файл project1.lpr с некоторыми приятными дополнениями, которые нужны почти всем программам. Перейдите к методу doRun
.
Проверка параметра
С TCustomApplication вы можете получить доступ к параметрам по имени. Например, ваша программа должна выводить текст справки, когда пользователь задает общий параметр справки -h. -h является коротким вариантом параметра. Длинная форма --help. Чтобы проверить, вызвал ли пользователь программу с помощью -h или --help, вы можете использовать следующий код:
if hasOption('h', 'help') then
begin
writeHelp;
halt;
end;
application.
перед hasOption
.Например:
if application.hasOption('h', 'help') then
begin
writeHelp;
halt;
end;
Если вы хотите поддерживать только короткий вариант, используйте:
if hasOption('h', '') then
Если вы хотите поддерживать только длинный вариант, используйте:
if hasOption('help') then
Чтение значения параметра
Каждому параметру может быть присвоено значение. Например:
$ project1 -f filename
или в длинной форме:
$ project1 --file=filename
Чтобы получить filename
, используйте:
writeLn('f=', getOptionValue('f', 'file'));
Option at position 1 needs an argument : f.
" (Переключатель в позиции 1 нуждается в аргументе: f), то вы забыли добавить опцию при вызове checkOptions
.Проверка параметров на валидность
Параметры командной строки представляют собой произвольный текст, поэтому пользователь при вводе может легко допускать ошибки. Отсюда проверка синтаксиса параметров обязательна. Для этого вы можете использовать метод CheckOptions.
Вы можете определить, какие параметры разрешены, какие из них нуждаются в параметре, а в случае синтаксической ошибки вы можете получить сообщение об ошибке, а также неправильные параметры для печати полезных и подробных ошибок.
Пример:
errorMsg := checkOptions('hf:', 'help file:');
Это позволяет передавать короткие переключатели: ‑f value
и ‑h
.
Это - передавать длинные переключатели: ‑‑help
или ‑‑file=filename
.
А это не позволяет передачу параметра ‑‑help
со значением или ‑‑file
без значения.
Пример передачи параметра:
procedure TMainForm.FormShow(Sender: TObject);
var
I: Integer;
Params : TStringList
LongOpts : array [1..2] of string = ('debug-sync', 'config-dir:');
begin
Params := TStringList.Create;
try
Application.GetNonOptions('hgo:', LongOpts, Params);
for I := 0 to Params.Count -1 do
debugln('Extra Param ' + inttostr(I) + ' is ' + Params[I]); }
finally
FreeAndNil(Params);
end;
end;
Этот пример находит параметры и не смешивает их с переключателями или параметрами командной строки. В этом примере приложение также принимает параметры --debug-sync и --config-dir=somedir, но здесь они не указаны.
Обратите внимание, что Application.GetNonOptions() принимает длинные параметры в виде массива, а Application.CheckOptions принимает те же данные, но в виде строки. Немного грустно!
Переменные окружения
Модуль sysUtils
определяет три основные функции для получения переменных среды.
getEnvironmentVariableCount
возвращает общее количество переменных среды.getEnvironmentString
возвращает переменную среды по ее индексуgetEnvironmentVariable
возвращает переменную среды по ее ключу
Пример:
program environmentVariablesList(input, output, stdErr);
uses
sysUtils;
var
i: integer;
begin
for i := 0 to getEnvironmentVariableCount() - 1 do
begin
writeLn(getEnvironmentString(i));
end;
end.
Также можно загрузить все переменные среды в объект TStringList
и легко получить доступ к паре имя-значение. Списки строк автоматически учитывают символ-разделитель, который по умолчанию является знаком равенства ('='
).
program environmentVariablesSorted(input, output, stdErr);
{$mode objFPC}
uses
classes, sysUtils;
var
i: integer;
environmentVariables: tStringList;
begin
environmentVariables := tStringList.create();
try
// загружаем все переменные в список строк
for i := 0 to getEnvironmentVariableCount() - 1 do
begin
environmentVariables.append(getEnvironmentString(i));
end;
environmentVariables.sort;
for i := 0 to environmentVariables.count - 1 do
begin
writeLn(environmentVariables.names[i]:20,
' = ',
environmentVariables.valueFromIndex[i]);
end;
finally
environmentVariables.free;
end;
end.
Существует также метод TCustomApplication
для одновременного получения всех переменных среды в виде TStringList
. Тогда схема может выглядеть так:
var
environmentVariables: tStringList;
begin
environmentVariables := tStringList.create();
try
application.getEnvironmentList(environmentVariables);
…
finally
environmentVariables.free;
end;
end;