Глава 24. Списки команд


Средством обработки последовательности из нескольких команд служат списки: "И-списки" и "ИЛИ-списки". Они эффективно могут заменить сложную последовательность вложенных if/then или даже case.

Объединение команд в цепочки

И-список
  1. command-1 && command-2 && command-3 && ... command-n
Каждая последующая команда, в таком списке, выполняется только тогда, когда предыдущая команда вернула код завершения true (ноль). Если какая-либо из команд возвращает false (не ноль), то исполнение списка команд в этом месте завершается, т.е. следующие далее команды не выполняются.

Пример 24-1. Проверка аргументов командной строки с помощью "И-списка"

  1. #!/bin/bash
  2. # "И-список"
  3. if [ ! -z "$1" ] && echo "Аргумент #1 = $1" && [ ! -z "$2" ] && echo "Аргумент #2 = $2"
  4. then
  5.   echo "Сценарию передано не менее 2 аргументов."
  6.   # Все команды в цепочке возвращают true.
  7. else
  8.   echo "Сценарию передано менее 2 аргументов."
  9.   # Одна из команд в списке вернула false.
  10. fi
  11. # Обратите внимание: "if [ ! -z $1 ]" тоже работает, но, казалось бы эквивалентный вариант
  12. #  if [ -n $1 ] — нет. Однако, если добавить кавычки
  13. #  if [ -n "$1" ] то все работает.  Будьте внимательны!
  14. # Проверяемые переменные лучше всегда заключать в кавычки.
  15. # То же самое, только без списка команд.
  16. if [ ! -z "$1" ]
  17. then
  18.   echo "Аргумент #1 = $1"
  19. fi
  20. if [ ! -z "$2" ]
  21. then
  22.   echo "Аргумент #2 = $2"
  23.   echo "Сценарию передано не менее 2 аргументов."
  24. else
  25.   echo "Сценарию передано менее 2 аргументов."
  26. fi
  27. # Получилось менее элегантно и длиннее, чем с использованием "И-списка".
  28. exit 0

Пример 24-2. Еще один пример проверки аргументов с помощью "И-списков"

  1. #!/bin/bash
  2. ARGS=1        # Ожидаемое число аргументов.
  3. E_BADARGS=65  # Код завершения, если число аргументов меньше ожидаемого.
  4. test $# -ne $ARGS && echo "Порядок использования: `basename $0` $ARGS аргумент(а)(ов)" && exit $E_BADARGS
  5. # Если проверка первого условия возвращает true (неверное число аргументов),
  6. # то исполняется остальная часть строки, и сценарий завершается.
  7. # Строка ниже выполняется только тогда, когда проверка выше не проходит.
  8. # обратите внимание на условие "-ne" — "не равно" (прим. перев.)
  9. echo "Сценарию передано корректное число аргументов."
  10. exit 0
  11. # Проверьте код завершения сценария командой "echo $?".

Конечно же, с помощью И-списка можно присваивать переменным значения по-умолчанию.

  1. arg1=$@       # В $arg1 записать аргументы командной строки.
  2. [ -z "$arg1" ] && arg1=DEFAULT
  3.               # Записать DEFAULT, если аргументы командной строки отсутствуют.


ИЛИ-список
  1. command-1 || command-2 || command-3 || ... command-n
Каждая последующая команда, в таком списке, выполняется только тогда, когда предыдущая команда вернула код завершения false (не ноль). Если какая-либо из команд возвращает true (ноль), то исполнение списка команд в этом месте завершается, т.е. следующие далее команды не выполняются. Очевидно, что "ИЛИ-списки" имеют смысл обратный, по отношению к "И-спискам"

Пример 24-3. Комбинирование "ИЛИ-списков" и "И-списков"

  1. #!/bin/bash
  2. #  delete.sh, утилита удаления файлов.
  3. #  Порядок использования: delete имя_файла
  4. E_BADARGS=65
  5. if [ -z "$1" ]
  6. then
  7.   echo "Порядок использования: `basename $0` имя_файла"
  8.   exit $E_BADARGS  # Если не задано имя файла.
  9. else
  10.   file=$1          # Запомнить имя файла.
  11. fi
  12. [ ! -f "$file" ] && echo "Файл \"$file\" не найден. \
  13. Робкий отказ удаления несуществующего файла."
  14. # И-СПИСОК, выдать сообщение об ошибке, если файл не существует.
  15. # Обратите внимание: выводимое сообщение продолжается во второй строке,
  16. # благодаря экранированию символа перевода строки.
  17. [ ! -f "$file" ] || (rm -f $file; echo "Файл \"$file\" удален.")
  18. # ИЛИ-СПИСОК, удаляет существующий файл.
  19. # Обратите внимание на логические условия.
  20. # И-СПИСОК отрабатывает по true, ИЛИ-СПИСОК — по false.
  21. exit 0
Important

Списки возвращают код завершения последней выполненной команды.

Комбинируя "И" и "ИЛИ" списки, легко "перемудрить" с логическими условиями, поэтому, в таких случаях может потребоваться детальная отладка.

  1. false && true || echo false         # false
  2. # Тот же результат дает
  3. ( false && true ) || echo false     # false
  4. # Но не эта комбинация
  5. false && ( true || echo false )     # (нет вывода на экран)
  6. #  Обратите внимание на группировку и порядок вычисления условий — слева-направо,
  7. #+ поскольку логические операции "&&" и "||" имеют равный приоритет.
  8. #  Если вы не уверены в своих действиях, то лучше избегать таких сложных конструкций.
  9. #  Спасибо S.C.


См. Пример A-8 и Пример 7-4, иллюстрирующие использование И/ИЛИ-списков для проверки переменных.