Выбрать главу

Код

Листинг 1.14. Сценарий valid-date

··#!/bin/bash

··# valid-date — Проверяет дату с учетом правил определения високосных лет

··normdate="укажите здесь имя файла, в котором вы сохранили сценарий normdate.sh"

··exceedsDaysInMonth()

··{

····# С учетом названия месяца и числа дней в этом месяце, данная функция

····# вернет: 0, если указанное число меньше или равно числу дней в месяце;

····# 1 — в противном случае.

····case $(echo $1|tr '[: upper: ]' '[: lower: ]') in

······jan*) days=31;; feb*) days=28;;

······mar*) days=31;; apr*) days=30;;

······may*) days=31;; jun*) days=30;;

······jul*) days=31;; aug*) days=31;;

······sep*) days=30;; oct*) days=31;;

······nov*) days=30;; dec*) days=31;;

········*) echo "$0: Unknown month name $1" >&2

············exit 1

····esac

····if [$2 −lt 1 −o $2 −gt $days]; then

······return 1

····else

······return 0 # Число месяца допустимо.

····fi

··}

··isLeapYear()

··{

····# Эта функция возвращает 0, если указанный год является високосным;

····#·· иначе возвращается 1.

····# Правила проверки високосного года:

····#·· 1. Если год не делится на 4, значит, он не високосный.

····#·· 2. Если год делится на 4 и на 400, значит, он високосный.

····#·· 3. Если год делится на 4, не делится на 400 и делится

····#······на 100, значит, он не високосный.

····#·· 4. Любой другой год, который делится на 4, является високосным.

····year=$1

····if ["$((year % 4))" −ne 0]; then

······return 1 # Nope, not a leap year.

····elif ["$((year % 400))" −eq 0]; then

······return 0 # Yes, it's a leap year.

····elif ["$((year % 100))" −eq 0]; then

······return 1

····else

······return 0

····fi

··}

··# Начало основного сценария

··# =================

··if [$# −ne 3]; then

····echo "Usage: $0 month day year" >&2

····echo "Typical input formats are August 3 1962 and 8 3 1962" >&2

····exit 1

··fi

··# Нормализовать дату и сохранить для проверки на ошибки.

··newdate="$($normdate "$@")"

··if [$? -eq 1]; then

····exit 1 # Error condition already reported by normdate

··fi

··# Разбить нормализованную дату, в которой

··#·· первое слово = месяц, второе слово = число месяца

··#·· третье слово = год.

··month="$(echo $newdate | cut −d\ −f1)"

··day="$(echo $newdate | cut −d\ −f2)"

··year="$(echo $newdate | cut −d\ −f3)"

··# После нормализации данных проверить допустимость

··#·· числа месяца (например, Jan 36 является недопустимой датой).

··if! exceedsDaysInMonth $month "$2"; then

····if ["$month" = "Feb" −a "$2" −eq "29"]; then

······if! isLeapYear $3; then

········echo "$0: $3 is not a leap year, so Feb doesn't have 29 days." >&2

········exit 1

······fi

····else

······echo "$0: bad day value: $month doesn't have $2 days." >&2

······exit 1

····fi

··fi

··echo "Valid date: $newdate"

··exit 0

Как это работает

Этот сценарий было очень интересно писать, потому что он требует проверки большого количества непростых условий: числа месяца, високосного года и так далее. Логика сценария не просто проверяет месяц как число от 1 до 12 или день — от 1 до 31. Чтобы сценарий проще было писать и читать, в нем используются специализированные функции.

Первая функция, exceedsDaysInMonth(), анализирует месяц, указанный пользователем, разрешая вероятные допущения (например, пользователь может передать название JANUAR, и оно будет правильно опознано). Анализ выполняется инструкцией case в строке , которая преобразует свой аргумент в нижний регистр и затем сравнивает полученное значение с константами, чтобы получить число дней в месяце. Единственный недостаток — для февраля функция всегда возвращает 28 дней.

Вторая функция, isLeapYear(), с помощью простых арифметических проверок выясняет, содержит ли февраль в указанном году 29-е число .

В основном сценарии исходные данные передаются сценарию normdate, представленному выше, для нормализации и затем разбиваются на три поля: $month, $day и $year. Затем вызывается функция exceedsDaysInMonth для проверки допустимости указанного числа для данного месяца, при этом 29 февраля обрабатывается отдельно — в этом случае вызовом функции isLeapYear проверяется год