Искусство программирования на языке сценариев командной оболочки
Шрифт:
Класс [:space:]– - соответствует пробельным символам (пробел и горизонтальная табуляция).
Класс [:upper:]– - соответствует набору символов алфавита в верхнем регистре. Эквивалентно выражению [A-Z].
Класс [:xdigit:]– - соответствует набору шестнадцатиричных цифр. Эквивалентно выражению [0-9A-Fa-f].
bash$ grep [[:digit:]] test.file
abc=723
Эти
bash$ ls -l ?[[:digit:]][[:digit:]]?
– rw-rw-r-- 1 bozo bozo 0 Aug 21 14:47 a33b
Примеры использования символьных классов в сценариях вы найдете в Пример 12-14 и Пример 12-15.
Sed, awk и Perl, используемые в сценариях в качестве фильтров, могут принимать регулярные выражения в качестве входных аргументов. См. Пример A-13 и Пример A-19.
Книга "Sed & Awk" (авторы Dougherty и Robbins) дает полное и ясное представление о регулярных выражениях (см. раздел Литература).
18.2. Globbing -- Подстановка имен файлов
Bash, сам по себе, не распознает регулярные выражения. Но в сценариях можно использовать команды и утилиты, такие как sed и awk, которые прекрасно справляются с обработкой регулярных выражений.
Фактически, Bash может выполнять подстановку имен файлов, этот процесс называется "globbing", но при этом не используется стандартный набор регулярных выражений. Вместо этого, при выполнении подстановки имен файлов, производится распознавание и интерпретация шаблонных символов. В число интерпретируемых шаблонов входят символы * и ?, списки символов в квадратных скобках и некоторые специальные символы (например ^, используемый для выполнения операции отрицания). Применение шаблонных символов имеет ряд важных ограничений. Например, если имена файлов начинаются с точки (например так: .bashrc), то они не будут соответствовать шаблону, содержащему символ * [ 48 ] . Аналогично, символ ? в операции подстановки имен файлов имеет иной смысл, нежели в регулярных выражениях.
48
Подстановка таких имен файлов возможна, но только при условии, что символ точки будет явно присутствовать в шаблоне.
~/[.]bashrc # Не будет соответствовать имени ~/.bashrc
~/?bashrc # То же самое.
# Метасимволы не могут соответствовать символу точки при подстановке имен файлов.
~/.[b]ashrc # Имя ~./bashrc будет соответствовать данному шаблону
~/.ba?hrc # Аналогично.
~/.bashr* # Аналогично.
# Установка ключа "dotglob" отключает такое поведение интерпретатора.
# Спасибо S.C.
bash$ ls -l
total 2
– rw-rw-r-- 1 bozo bozo 0 Aug 6 18:42 a.1
– rw-rw-r-- 1 bozo bozo 0 Aug 6 18:42 b.1
– rw-rw-r-- 1 bozo bozo 0 Aug 6 18:42 c.1
– rw-rw-r-- 1 bozo bozo 466 Aug 6 17:48 t2.sh
– rw-rw-r-- 1 bozo bozo 758 Jul 30 09:02 test1.txt
bash$ ls -l t?.sh
– rw-rw-r-- 1 bozo bozo 466 Aug 6 17:48 t2.sh
bash$ ls -l [ab]*
– rw-rw-r-- 1 bozo bozo 0 Aug 6 18:42 a.1
– rw-rw-r-- 1 bozo bozo 0 Aug 6 18:42 b.1
bash$ ls -l [a-c]*
– rw-rw-r-- 1 bozo bozo 0 Aug 6 18:42 a.1
– rw-rw-r-- 1 bozo bozo 0 Aug 6 18:42 b.1
– rw-rw-r-- 1 bozo bozo 0 Aug 6 18:42 c.1
bash$ ls -l [^ab]*
– rw-rw-r-- 1 bozo bozo 0 Aug 6 18:42 c.1
– rw-rw-r-- 1 bozo bozo 466 Aug 6 17:48 t2.sh
– rw-rw-r-- 1 bozo bozo 758 Jul 30 09:02 test1.txt
bash$ ls -l {b*,c*,*est*}
– rw-rw-r-- 1 bozo bozo 0 Aug 6 18:42 b.1
– rw-rw-r-- 1 bozo bozo 0 Aug 6 18:42 c.1
– rw-rw-r-- 1 bozo bozo 758 Jul 30 09:02 test1.txt
bash$ echo *
a.1 b.1 c.1 t2.sh test1.txt
bash$ echo t*
t2.sh test1.txt
Даже
См. также Пример 10-4.
Глава 19. Подоболочки, или Subshells
Запуск сценария приводит к запуску дочернего командного интерпретатора. Который выполняет интерпретацию и исполнение списка команд, содержащихся в файле сценария, точно так же, как если бы они были введены из командной строки. Любой сценарий запускается как дочерний процесс родительской командной оболочки, той самой, которая выводит перед вами строку приглашения к вводу на консоли или в окне xterm.
Сценарий может, так же, запустить другой дочерний процесс, в своей подоболочке. Это позволяет сценариям распараллелить процесс обработки данных по нескольким задачам, исполняемым одновременно.
Список команд в круглых скобках
( command1; command2; command3; ... )
Список команд, в круглых скобках, исполняется в подоболочке.
Пример 19-1. Область видимости переменных
#!/bin/bash
# subshell.sh
echo
outer_variable=Outer
(
inner_variable=Inner
echo "Дочерний процесс, \"inner_variable\" = $inner_variable"
echo "Дочерний процесс, \"outer\" = $outer_variable"
)
echo
if [ -z "$inner_variable" ]
then
echo "Переменная inner_variable не определена в родительской оболочке"
else
echo "Переменная inner_variable определена в родительской оболочке"
fi
echo "Родительский процесс, \"inner_variable\" = $inner_variable"
# Переменная $inner_variable не будет определена
# потому, что переменные, определенные в дочернем процессе,
# ведут себя как "локальные переменные".
echo
exit 0
См. также Пример 31-1.
+
Смена текущего каталога в дочернем процессе (подоболочке) не влечет за собой смену текущего каталога в родительской оболочке.