$ ./my_name Rick Neil
Original parameters are Rick Neil
Is your name Rick ?
Enter yes or no: yes
Hi Rick, nice name
$
Как это работает
Когда сценарий начинает выполняться, функция определена, но еще не выполняется. В операторе if
сценарий вызывает функцию yes_or_no
, передавая ей оставшуюся часть строки как параметры после замены $1
первым параметром исходного сценария строкой Rick
. Функция использует эти параметры, в данный момент хранящиеся в позиционных параметрах $1
, $2
и т.д., и возвращает значение в вызывающую программу. В зависимости от возвращенного функцией значения конструкция if
выполняет один из операторов.
Как видите, у командной оболочки есть большой набор управляющих структур и условных операторов. Вам необходимо познакомиться с некоторыми командами, встроенными в оболочку; после этого вы будете готовы решать реальные программистские задачи без компилятора под рукой!
Команды
В сценариях командной оболочки можно выполнять два сорта команд. Как уже упоминалось, существуют "обычные" команды, которые могут выполняться и из командной строки (называемые внешними командами), и встроенные команды (называемые внутренними командами). Внутренние команды реализованы внутри оболочки и не могут вызываться как внешние программы. Но большинство внутренних команд представлено и в виде автономных программ, это условие — часть требований стандарта POSIX. Обычно, не важно, команда внешняя или внутренняя, за исключением того, что внутренние команды действуют эффективнее.
В этом разделе представлены основные команды, как внутренние, так и внешние, которые мы используем при написании сценариев. Как пользователь ОС Linux, вы, возможно, знаете много других команд, которые принимает командная строка. Всегда помните о том, что вы можете любую из них применить в сценарии в дополнение к встроенным командам, представленным в данном разделе.
Используйте команду break
для выхода из циклов for
, while
и until
до того, как будет удовлетворено управляющее условие. В команде break
можно задать дополнительный числовой параметр, указывающий на число циклов, из которых предполагается выход. Однако это может сильно усложнить чтение сценариев, поэтому мы не советуем вам использовать его. По умолчанию break
обеспечивает выход из одного цикла.
#!/bin/sh
rm -rf fred*
echo > fred1
echo > fred2
mkdir fred3
echo > fred4
for file in fred*
do
if [ -d "$file" ]; then
break;
fi
done
echo first directory starting fred was $file
m -rf fred*
exit 0
Команда "двоеточие" — фиктивная команда. Она иногда полезна для упрощения логики в условиях, будучи псевдонимом команды true
. Поскольку команда :
встроенная, она выполняется быстрее, чем true
, хотя ее вывод гораздо менее читабелен.
Вы можете найти эту команду в условии для циклов while
. Конструкция while :
выполняет бесконечный цикл вместо более общего while true
.
Конструкция :
также полезна для условного задания переменных. Например,
: ${var:=value}
Без :
командная оболочка попытается интерпретировать $var
как команду.
В некоторых более старых версиях сценариев командной оболочки можно встретить двоеточие, применяемое в начале строки для обозначения комментариев, однако в современных сценариях следует всегда применять для обозначения начала комментариев знак #
, поскольку этот вариант действует эффективнее.
#!/bin/sh
rm -f fred
if [ -f fred ]; then
:
else
echo file fred did not exist
fi
exit 0
Как и одноименный оператор языка С, эта команда заставляет охватывающий ее цикл for
, while
или until
начать новый проход или следующую итерацию. При этом переменная цикла принимает следующее значение в списке.
#!/bin/sh
rm -rf fred*
echo > fred1
echo > fred2
mkdir fred3
echo > fred4
for file in fred*
do
if [ -d "$file" ]; then
echo "skipping directory $file"
continue
fi
echo file is $file
done
rm -rf fred*
exit 0
Команда continue
может принимать в качестве необязательного параметра номер прохода охватывающего цикла, с которого следует возобновить выполнение цикла.
Таким образом, вы сможете иногда выскочить из вложенных циклов. Данный параметр редко применяется, т.к. он часто сильно затрудняет понимание сценариев. Например,
for x in 1 2 3
do
echo before $x
continue 1
echo after $x
done
У приведенного фрагмента будет следующий вывод: