KnigaRead.com/
KnigaRead.com » Компьютеры и Интернет » Программирование » Мендель Купер - Искусство программирования на языке сценариев командной оболочки

Мендель Купер - Искусство программирования на языке сценариев командной оболочки

На нашем сайте KnigaRead.com Вы можете абсолютно бесплатно читать книгу онлайн Мендель Купер, "Искусство программирования на языке сценариев командной оболочки" бесплатно, без регистрации.
Перейти на страницу:

И-список

command-1 && command-2 && command-3 && ... command-n

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

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

#!/bin/bash

# "И-список"


if [ ! -z "$1" ] && echo "Аргумент #1 = $1" && [ ! -z "$2" ] && echo "Аргумент #2 = $2"

then

echo "Сценарию передано не менее 2 аргументов."

# Все команды в цепочке возвращают true.

else

echo "Сценарию передано менее 2 аргументов."

# Одна из команд в списке вернула false.

fi

# Обратите внимание: "if [ ! -z $1 ]" тоже работает, но, казалось бы эквивалентный вариант

# if [ -n $1 ] -- нет. Однако, если добавить кавычки

# if [ -n "$1" ] то все работает. Будьте внимательны!

# Проверяемые переменные лучше всегда заключать в кавычки.


# То же самое, только без списка команд.

if [ ! -z "$1" ]

then

echo "Аргумент #1 = $1"

fi

if [ ! -z "$2" ]

then

echo "Аргумент #2 = $2"

echo "Сценарию передано не менее 2 аргументов."

else

echo "Сценарию передано менее 2 аргументов."

fi

# Получилось менее элегантно и длиннее, чем с использованием "И-списка".


exit 0

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

#!/bin/bash


ARGS=1 # Ожидаемое число аргументов.

E_BADARGS=65 # Код завершения, если число аргументов меньше ожидаемого.


test $# -ne $ARGS && echo "Порядок использования: `basename $0` $ARGS аргумент(а)(ов)" && exit $E_BADARGS

# Если проверка первого условия возвращает true (неверное число аргументов),

# то исполняется остальная часть строки, и сценарий завершается.


# Строка ниже выполняется только тогда, когда проверка выше не проходит.

# обратите внимание на условие "-ne" -- "не равно" (прим. перев.)

echo "Сценарию передано корректное число аргументов."


exit 0


# Проверьте код завершения сценария командой "echo $?".

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

[email protected] # В $arg1 записать аргументы командной строки.


[ -z "$arg1" ] && arg1=DEFAULT

# Записать DEFAULT, если аргументы командной строки отсутствуют.


ИЛИ-список

command-1 || command-2 || command-3 || ... command-n

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

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

#!/bin/bash


# delete.sh, утилита удаления файлов.

# Порядок использования: delete имя_файла


E_BADARGS=65


if [ -z "$1" ]

then

echo "Порядок использования: `basename $0` имя_файла"

exit $E_BADARGS # Если не задано имя файла.

else

file=$1 # Запомнить имя файла.

fi


[ ! -f "$file" ] && echo "Файл "$file" не найден.

Робкий отказ удаления несуществующего файла."

# И-СПИСОК, выдать сообщение об ошибке, если файл не существует.

# Обратите внимание: выводимое сообщение продолжается во второй строке,

# благодаря экранированию символа перевода строки.


[ ! -f "$file" ] || (rm -f $file; echo "Файл "$file" удален.")

# ИЛИ-СПИСОК, удаляет существующий файл.


# Обратите внимание на логические условия.

# И-СПИСОК отрабатывает по true, ИЛИ-СПИСОК -- по false.


exit 0

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

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

false && true || echo false # false


# Тот же результат дает

( false && true ) || echo false # false

# Но не эта комбинация

false && ( true || echo false ) # (нет вывода на экран)


# Обратите внимание на группировку и порядок вычисления условий -- слева-направо,

#+ поскольку логические операции "&&" и "||" имеют равный приоритет.


# Если вы не уверены в своих действиях, то лучше избегать таких сложных конструкций.


# Спасибо S.C.


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


Глава 25. Массивы

Новейшие версии Bash поддерживают одномерные массивы. Инициализация элементов массива может быть произведена в виде: variable[xx]. Можно явно объявить массив в сценарии, с помощью директивы declare: declare -a variable. Обращаться к отдельным элементам массива можно с помощью фигурных скобок, т.е.: ${variable[xx]}.

Пример 25-1. Простой массив

#!/bin/bash


area[11]=23

area[13]=37

area[51]=UFOs


# Массивы не требуют, чтобы последовательность элементов в массиве была непрерывной.


# Некоторые элементы массива могут оставаться неинициализированными.

# "Дыркм" в массиве не являются ошибкой.


echo -n "area[11] = "

echo ${area[11]} # необходимы {фигурные скобки}


echo -n "area[13] = "

echo ${area[13]}


echo "содержимое area[51] = ${area[51]}."


# Обращение к неинициализированным элементам дает пустую строку.

echo -n "area[43] = "

echo ${area[43]}

echo "(элемент area[43] -- неинициализирован)"


echo


# Сумма двух элементов массива, записанная в третий элемент

area[5]=`expr ${area[11]} + ${area[13]}`

echo "area[5] = area[11] + area[13]"

echo -n "area[5] = "

echo ${area[5]}


area[6]=`expr ${area[11]} + ${area[51]}`

echo "area[6] = area[11] + area[51]"

echo -n "area[6] = "

echo ${area[6]}

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


echo; echo; echo


# -----------------------------------------------------------------

# Другой массив, "area2".

# И другой способ инициализации массива...

# array_name=( XXX YYY ZZZ ... )


area2=( ноль один два три четыре )


echo -n "area2[0] = "

echo ${area2[0]}

# Ага, индексация начинается с нуля (первый элемент массива имеет индекс [0], а не [1]).


echo -n "area2[1] = "

echo ${area2[1]} # [1] -- второй элемент массива.

# -----------------------------------------------------------------


echo; echo; echo


# -----------------------------------------------

# Еще один массив, "area3".

# И еще один способ инициализации...

# array_name=([xx]=XXX [yy]=YYY ...)


area3=([17]=семнадцать [21]=двадцать_один)


echo -n "area3[17] = "

echo ${area3[17]}


echo -n "area3[21] = "

echo ${area3[21]}

# -----------------------------------------------


exit 0

Bash позволяет оперировать переменными, как массивами, даже если они не были явно объявлены таковыми.

string=abcABC123ABCabc

echo ${string[@]} # abcABC123ABCabc

echo ${string[*]} # abcABC123ABCabc

echo ${string[0]} # abcABC123ABCabc

echo ${string[1]} # Ничего не выводится!

# Почему?

echo ${#string[@]} # 1

# Количество элементов в массиве.


# Спасибо Michael Zick за этот пример.

Эти примеры еще раз подтверждают отсутствие контроля типов в Bash.

Пример 25-2. Форматирование стихотворения

#!/bin/bash

# poem.sh


# Строки из стихотворения (одна строфа).

Line[1]="Мой дядя самых честных правил,"

Line[2]="Когда не в шутку занемог;"

Line[3]="Он уважать себя заставил,"

Line[4]="И лучше выдумать не мог."

Line[5]="Его пример другим наука..."


# Атрибуты.

Attrib[1]=" А.С. Пушкин"

Attrib[2]=""Евгений Онегин""


for index in 1 2 3 4 5 # Пять строк.

do

printf " %sn" "${Line[index]}"

done


for index in 1 2 # Две строки дополнительных атрибутов.

do

printf " %sn" "${Attrib[index]}"

done


exit 0

При работе с отдельными элементами массива можно использовать специфический синтаксис, даже стандартные команды и операторы Bash адаптированы для работы с массивами.

array=( ноль один два три четыре пять )


echo ${array[0]} # ноль

echo ${array:0} # ноль

# Подстановка параметра -- первого элемента.

echo ${array:1} # оль

# Подстановка параметра -- первого элемента,

#+ начиная с позиции #1 (со 2-го символа).


echo ${#array} # 4

# Длина первого элемента массива.


array2=( [0]="первый элемент" [1]="второй элемент" [3]="четвертый элемент" )


echo ${array2[0]} # первый элемент

echo ${array2[1]} # второй элемент

echo ${array2[2]} #

# Элемент неинициализирован, поэтому на экран ничего не выводится.

echo ${array2[3]} # четвертый элемент


При работе с массивами, некоторые встроенные команды Bash имеют несколько иной смысл. Например, unset -- удаляет отдельные элементы массива, или даже массив целиком.

Пример 25-3. Некоторые специфичные особенности массивов

#!/bin/bash

Перейти на страницу:
Прокомментировать
Подтвердите что вы не робот:*