Программирование на языке Shell в системе UNIX. Примеры программирования
1. ВВЕДЕНИЕ
Среди операционных систем особое место занимает Unix. Беспрецедентным является
то, что ОС Unix может работать практически на всех выпускаемых платформах. UNIX
- это стандарт де факто открытых и мобильных операционных систем. (поскольку
название UNIX запатентовано компанией AT&T - различные юниксы называются
различно: SCO UNIX, BSDI, Solaris, Linux, DG/UX, AIX и т.д.).
Это не только многозадачная, но и многопользовательская система. Она
обеспечивает современный пользовательский интерфейс на базе системы X Window и
межмашинную связь на базе протоколов TCP/IP и т.п.
ОС Unix была создана Кеном Томпсоном и Деннисом Ритчи в Bell Laborotories
(AT&T). Широко распространяться Unix/v7 (версия 7) начала в 79 - 80-м годах.
Вручение создателям Unix в 1983 году Международной премии А.Тьюринга в области
программирования ознаменовало признание этой системы мировой научной (computer
science) общественностью. Что также беспрецедентно.
ОС Unix стоит на трех китах: язык Си, файловая система, командный язык. В
дальнейшем к ним добавились система X Window и протоколы TCP/IP.
2. ПРОСТЕЙШИЕ СРЕДСТВА SHELL
Командный язык shell (в переводе - раковина, скорлупа) фактически есть язык
программирования очень высокого уровня. На этом языке пользователь осуществляет
управление компьютером. Обычно, после входа в систему вы начинаете
взаимодействовать с командной оболочкой. Признаком того, что оболочка (shell)
готова к приему команд служит выдаваемый ею на экран промптер. В простейшем
случае это один доллар ("$").
Shell не является необходимым и единственным командным языком (хотя именно он
стандартизован в рамках POSIX [POSIX 1003.2] - стандарта мобильных систем).
Например, немалой популярностью пользуется язык cshell, есть также kshell,
bashell и другие. Более того, каждый пользователь может создать свой командный
язык. Может одновременно на одном экземпляре операционной системы работать с
разными командными языками.
shell - это одна из многих команд UNIX. То есть в набор команд оболочки "shell"
входит команда "sh" - вызов интерпретатора "shell". Первый "shell" вызывается
автоматически при вашем входе в систему и выдает на экран промтер. После этого
вы можете вызывать на выполнение любые команды, в том числе и снова сам "shell",
который вам создаст новую оболочку внутри прежней.
Так например, если вы подготовите в редакторе файл "file_1":
echo Hello!
то это будет обычный текстовый файл, содержащий команду "echo", которая при
выполнении выдает все написанное правее ее на экран. Можно сделать файл "file_1"
выполняемым с помощью команды "chmod 755 file_1". Но его можно выполнить, вызвав
явно команду "sh" ("shell"):
sh file_1
или
sh < file1
Файл можно выполнить и в текущем экземпляре "shell". Для этого существует
специфическая команда "." (точка), т.е.
. file_1
Поскольку UNIX - система многопользовательская, вы можете даже на персональном
компьютере работать параллельно, скажем, на 12-ти экранах (переход с экрана на
экран ALT/функциональная клавиша), имея на каждом экране нового (или одного и
того же) пользователя со своей командной оболочкой. Можете и в графическом
режиме X-Window также открыть большое число окон, а в каждом окне может быть
свой пользователь со своей командной оболочкой...
Стержневым элементом языка shell является команда.
2.1. Структура команд
Команды в shell обычно имеют следующий формат:
<имя команды> <флаги> <аргумент(ы)>
Например:
ls -ls /usr/bin
Где ls - имя команды выдачи содержимого директория,
-ls - флаги ( "-" - признак флагов, l - длинный формат, s - об'ем файлов в
блоках),
/usr/bin - директорий, для которого выполняется команда.
Эта команда выдаст на экран в длинном формате содержимое директория /usr/bin,
при этом добавит информацию о размере каждого файла в блоках.
К сожалению, такая структура команды выдерживается далеко не всегда. Не всегда
перед флагами ставится минус, не всегда флаги идут одним словом. Есть
разнообразие и в представлении аргументов. К числу команд, имеющих экзотические
форматы, относятся и такие "ходовые" команды, как сс – вызов компилятора языка
С, tar – работа с архивами, dd – копирование файла с преобразованием, find –
поиск файлов и ряд других.
Как правило, первое слово shell воспринимает, как команду. Поэтому в командной
строке
cat cat
первое слово будет расшифровано shell, как команда (конкатенации), которая
выдаст на экран файл с именем "cat" (второе слово), находящийся в текущем
директории.
2.2. Группировка команд.
Средства группировки:
; и <перевод строки> - определяют последовательное выполнение команд;
& - асинхронное (фоновое) выполнение предшествующей команды;
&& - выполнение последующей команды при условии нормального завершения
предыдущей, иначе игнорировать;
- выполнение последующей команды при ненормальном завершении предыдущей, иначе
игнорировать.
При выполнении команды в асинхронном режиме (после команды стоит один амперсанд)
на экран выводится номер процесса, соответствующий выполняемой команде, и
система, запустив этот фоновый процесс, вновь выходит на диалог с пользователем.
Например, наберем команду "find" в фоновом режиме для поиска в системе , начиная
от корня "/", файла с именем "conf", а затем "pwd" в обычном режиме. На экране
этот фрагмент будет выглядеть следующим образом (курсивом выделены комментарии):
$ find/-name conf -print & ввод команды "find"
288 номер (PID) фонового процесса
$ pwd ввод команды "pwd"
/mnt/lab/asu результат работы "pwd"
$ возвращение shell в промптер
/usr/include/sys/conf результат работы "find"
Иногда необходимо, чтобы все фоновые процессы завершились, прежде чем будет
выполняться какой-то расчет. Для этого служит специальная команда "wait [PID]".
Эта команда ждет завершения указанного идентификатором (числом) фонового
процесса. Если команда без параметра, то она ждет завершения всех фоновых
процессов, дочерних для данного "sh".
Для группировки команд также могут использоваться фигурные "{}" и круглые "()"
скобки. Рассмотрим примеры, сочетающие различные способы группировки: Если
введена командная строка
command1 && command2; command3
где command1, command2 и command3 - какие-то команды, то "command2" будет
выполнена только при успешном завершении "command1"; после любого из исходов
обработки "command2" (т.е. "command2" будет выполнена, либо пропущена) будет
выполнена "command3".
command1 && { command2; command3}
Здесь обе команды ("command2" и "command3") будут выполнены только при успешном
завершении "command1".
{command1; command2} &
В фоновом режиме будет выполняться последовательность команд "command1" и
"command2".
Фоновые процессы сложно уничтожить, поскольку традиционная команда "CTL/C"
прерывает только процессы переднего плана. Для уничтожения фонового процесса
надо знать его номер. При запуске фонового процесса на экран выдается число,
соответствующее номеру (идентификатору) этого процесса (PID). Если этот номер
забыт или надо убедиться, что этот процесс не закончен, с помощью команды
ps -aux
можно получить перечень идентификаторов процессов (PID), имена пользователей,
текущее время, затраченное процессами, и т.д.
В выведенной таблице можно найти номера процессов, подлежащих уничтожению,
например это "849" и "866". Тогда командой
kill -9 866 849
можно уничтожить эти процессы. При уничтожении процессов надо вы должны иметь то
же имя пользователя, какое было приписано уничтожаемым процессам или root.
Круглые скобки "()", кроме выполнения функции группировки, выполняют и функцию
вызова нового экземпляра интерпретатора shell.
Пусть мы находились в начальном каталоге "/mnt/lab/asu"
Тогда в последовательности команд
cd ..; ls; ls
две команды "ls" выдадут 2 экземпляра содержимого каталога "/mnt/lab", а
последовательность
(cd ..; ls) ls
выдаст сначала содержимое каталога "/mnt/lab", а затем содержимое
"/mnt/lab/asu", т.к. при входе в скобки вызывается новый экземпляр shell, в
рамках которого и осуществляется переход. При выходе из круглых скобок
происходит возврат в старый shell и в старый каталог.
2.3. Перенаправление команд
Стандартный ввод (вход) - "stdin" в ОС UNIX осуществляется с клавиатуры
терминала, а стандартный вывод (выход) - "stdout" направлен на экран терминала.
Существует еще и стандартный файл диагностических сообщений - "stderr", о
котором речь будет чуть позже.
Команда, которая может работать со стандартным входом и выходом, называется
ФИЛЬТРОМ.
Пользователь имеет удобные средства перенаправления ввода и вывода на другие
файлы (устройства). Символы ">" и ">>" обозначают перенаправление вывода.
ls >file_1
команда "ls" сформирует список файлов текущего каталога и поместит его в файл
"file_1" (вместо выдачи на экран). Если файл "file_1" до этого существовал, то
он будет затерт новым.
pwd >>file_1
команда pwd сформирует полное имя текущего каталога и поместит его в конец файла
"file_1", т.е. ">>" добавляет в файл, если он непустой.
Символы "<" и "<<" обозначают перенаправление ввода.
wc -l <file_1
подсчитает и выдаст на экран число строк в файле file_1.
ed file_2 <<!
создаст с использованием редактора файл "file_2", непосредственно с терминала.
Окончание ввода определяется по символу, стоящему правее "<<" (т. е. "!"). То
есть ввод будет закончен, когда первым в очередной строке будет "!".
Можно сочетать перенаправления. Так
wc -l <file_3 >file_4
и
wc -l >file_4 <file_3
выполняются одинаково: подсчитывается число строк файла "file_3" и результат