Глава 19. Подоболочки, или Subshells | Популярный Linux

Опубликовано Bash в Ср, 23/12/2009 — 21:04


Запуск сценария приводит к запуску дочернего командного интерпретатора. Который выполняет интерпретацию и исполнение списка команд, содержащихся в файле сценария, точно так же, как если бы они были введены из командной строки. Любой сценарий запускается как дочерний процесс родительской командной оболочки, той самой, которая выводит перед вами строку приглашения к вводу на консоли или в окне xterm.

Сценарий может, так же, запустить другой дочерний процесс, в своей подоболочке. Это позволяет сценариям распараллелить процесс обработки данных по нескольким задачам, исполняемым одновременно.

Как правило, внешние команды в сценариях порождают дочерние процессы, в то время как встроенные команды Bash — нет. По этой причине, встроенные команды выполняются много быстреесвоих «внешних» аналогов.

Список команд в круглых скобках

( command1; command2; command3; … )

Список команд, в круглых скобках, исполняется в подоболочке.

Note

Значения переменных, определенных в дочерней оболочке, не могут быть переданы родительской оболочке. Они недоступны родительскому процессу. Фактически, они ведут себя как локальные переменные.

Пример 19-1. Область видимости переменных

  1. #!/bin/bash
  2. # subshell.sh
  3. echo
  4. outer_variable=Outer
  5. (
  6. inner_variable=Inner
  7. echo «Дочерний процесс, inner_variable = $inner_variable«
  8. echo «Дочерний процесс, outer = $outer_variable«
  9. )
  10. echo
  11. if [ -z «$inner_variable« ]
  12. then
  13.   echo «Переменная inner_variable не определена в родительской оболочке»
  14. else
  15.   echo «Переменная inner_variable определена в родительской оболочке»
  16. fi
  17. echo «Родительский процесс, inner_variable = $inner_variable«
  18. # Переменная $inner_variable не будет определена
  19. # потому, что переменные, определенные в дочернем процессе,
  20. # ведут себя как «локальные переменные».
  21. echo
  22. exit 0

См. также Пример 31-2.

+

Смена текущего каталога в дочернем процессе (подоболочке) не влечет за собой смену текущего каталога в родительской оболочке.

Пример 19-2. Личные настройки пользователей

  1. #!/bin/bash
  2. # allprofs.sh: вывод личных настроек (profiles) всех пользователей
  3. # Автор: Heiner Steven
  4. # С некоторыми изменениями, внесенными автором документа.
  5. FILE=.bashrc  #  Файл настроек пользователя,
  6.               #+ в оригинальном сценарии называется «.profile».
  7. for home in `awk -F: ‘{print $6}’ /etc/passwd`
  8. do
  9.   [ -d «$home« ] || continue    # Перейти к следующей итерации, если нет домашнего каталога.
  10.   [ -r «$home« ] || continue    # Перейти к следующей итерации, если не доступен для чтения.
  11.   (cd $home; [ -e $FILE ] && less $FILE)
  12. done
  13. #  По завершении сценария — нет теобходимости выполнять команду ‘cd’, чтобы вернуться в первоначальный каталог,
  14. #+ поскольку ‘cd $home’ выполняется в подоболочке.
  15. exit 0

Подоболочка может использоваться для задания «специфического окружения» для группы команд.

  1. COMMAND1
  2. COMMAND2
  3. COMMAND3
  4. (
  5.   IFS=:
  6.   PATH=/bin
  7.   unset TERMINFO
  8.   set -C
  9.   shift 5
  10.   COMMAND4
  11.   COMMAND5
  12.   exit 3 # Выход только из подоболочки.
  13. )
  14. # Изменение переменных окружения не коснется родительской оболочки.
  15. COMMAND6
  16. COMMAND7

Как вариант использования подоболочки — проверка переменных.

  1. if (set -u; : $variable) 2> /dev/null
  2. then
  3.   echo «Переменная определена.»
  4. fi     #  Переменная была инициализирована в данном сценарии
  5.        #+ или это внутренняя переменная Bash
  6.        #+ или это переменная окружения (была экспортирована в оболочку).
  7. # Можно сделать то же самое по другому: [[ ${variable-x} != x || ${variable-y} != y ]]
  8. # или                                   [[ ${variable-x} != x$variable ]]
  9. # или                                   [[ ${variable+x} = x ]])
  10. # или                                   [[ ${variable-x} != x ]])

Еще одно применение — проверка файлов блокировки:

  1. if (set -C; : > lock_file) 2> /dev/null
  2. then
  3.   :   # lock_file отсутствует: программа (сценарий) не запущена
  4. else
  5.   echo «Этот сценарий уже запущен другим пользователем.»
  6.   exit 65
  7. fi
  8. #  Автор фрагмента Stephane Chazelas,
  9. #+ небольшие дополнения сделаны Paulo Marcel Coelho Aragao.

Процессы в подоболочках могут исполняться параллельно. Это позволяет разбить сложную задачу на несколько простых подзадач, выполняющих параллельную обработку информации.

Пример 19-3. Запуск нескольких процессов в подоболочках

  1.   (cat list1 list2 list3 | sort | uniq > list123) &
  2.   (cat list4 list5 list6 | sort | uniq > list456) &
  3.   # Слияние и сортировка двух списков производится одновременно.
  4.   # Запуск в фоне гарантирует параллельное исполнение.
  5.   #
  6.   # Тот же эффект дает
  7.   #   cat list1 list2 list3 | sort | uniq > list123 &
  8.   #   cat list4 list5 list6 | sort | uniq > list456 &
  9.   wait   # Ожидание завершения работы подоболочек.
  10.   diff list123 list456

Перенаправление ввода/вывода в/из подоболочки производится оператором построения конвейера «|», например, ls -al | (command).

Note

Блок команд, заключенный в фигурные скобки не приводит к запуску дочерней подоболочки.

{ command1; command2; command3; … }

Запись опубликована в рубрике Без рубрики. Добавьте в закладки постоянную ссылку.