П О Р Т А Л                            
С Е Т Е В Ы Х                          
П Р О Е К Т О В                        
  
Поиск по сайту:
                                                 
Главная

О проекте

Web-мастеру
     HTML & JavaScript
     SSI
     Perl
     PHP
     XML & XSLT
     Unix Shell

MySQL

Безопасность

Хостинг

Другое








Самое читаемое:

Учебник PHP - "Для Чайника".
Просмотров 3979 раз(а).

Иллюстрированный самоучитель по созданию сайтов.
Просмотров 6579 раз(а).

Учебник HTML.
Просмотров 3609 раз(а).

Руководство по PHP5.
Просмотров 5872 раз(а).

Хостинг через призму DNS.
Просмотров 4652 раз(а).

Подборка текстов стандартных документов.
Просмотров 56128 раз(а).

Учебник PHP - Самоучитель
Просмотров 3539 раз(а).

Документация на MySQL (учебник & справочное руководство)
Просмотров 7876 раз(а).

Внешние атаки...
Просмотров 4345 раз(а).

Учебник PHP.
Просмотров 3092 раз(а).

SSI в примерах.
Просмотров 114 раз(а).



 
 
| Добавить в избранное | Сделать стартовой | Помощь





33.4. Рекурсия

Может ли сценарий рекурсивно вызывать себя самого? Да, может!

Пример 33-6. Сценарий (бесполезный), который вызывает себя сам

#!/bin/bash
# recurse.sh

#  Может ли сценарий вызвать себя сам?
#  Да, но есть ли в этом смысл?

RANGE=10
MAXVAL=9

i=$RANDOM
let "i %= $RANGE"  # Генерация псевдослучайного числа в диапазоне 0 .. $MAXVAL.

if [ "$i" -lt "$MAXVAL" ]
then
  echo "i = $i"
  ./$0             #  Сценарий запускает новый экземпляр себя самого.
fi                 #  если число $i больше или равно $MAXVAL.

#  Если конструкцию "if/then" заменить на цикл "while", то это вызовет определенные проблемы.
#  Объясните -- почему?.

exit 0

Пример 33-7. Сценарий (имеющий практическую ценность), который вызывает себя сам

#!/bin/bash
# pb.sh: телефонная книга

# Автор: Rick Boivie
# используется с его разрешения.
# Дополнен автором документа.

MINARGS=1     # Сценарию должен быть передан, по меньшей мере, один аргумент.
DATAFILE=./phonebook
PROGNAME=$0
E_NOARGS=70   # Ошибка, нет аргументов.

if [ $# -lt $MINARGS ]; then
      echo "Порядок использования: "$PROGNAME" data"
      exit $E_NOARGS
fi


if [ $# -eq $MINARGS ]; then
      grep $1 "$DATAFILE"
else
      ( shift; "$PROGNAME" $* ) | grep $1
      # Рекурсивный вызов.
fi

exit 0        #  Сценарий завершает свою работу здесь.
              #  Далее следует пример файла телефонной книги
              #+ в котором не используются символы комментария.

# ------------------------------------------------------------------------
# Пример файла телефонной книги

John Doe        1555 Main St., Baltimore, MD 21228          (410) 222-3333
Mary Moe        9899 Jones Blvd., Warren, NH 03787          (603) 898-3232
Richard Roe     856 E. 7th St., New York, NY 10009          (212) 333-4567
Sam Roe         956 E. 8th St., New York, NY 10009          (212) 444-5678
Zoe Zenobia     4481 N. Baker St., San Franciso, SF 94338   (415) 501-1631
# ------------------------------------------------------------------------

$bash pb.sh Roe
Richard Roe     856 E. 7th St., New York, NY 10009          (212) 333-4567
Sam Roe         956 E. 8th St., New York, NY 10009          (212) 444-5678

$bash pb.sh Roe Sam
Sam Roe         956 E. 8th St., New York, NY 10009          (212) 444-5678

#  Если сценарию передаются несколько аргументов,
#+ то выводятся только те строки, которые содержат их все.

Пример 33-8. Еще один сценарий, который вызывает себя сам

#!/bin/bash
# usrmnt.sh, автор Anthony Richardson
# Используется с его разрешения.

# Порядок использования:  usrmnt.sh
# Описание: монтирует устройство, пользователь должен входить в состав группы
#              MNTUSERS в файле /etc/sudoers.

# -----------------------------------------------------------------
#  Этот сценарий рекурсивно вызывает себя самого,
#+ используя sudo. Пользователь, наделенный
#+ соответствующими правами может просто дать команду

#   usermount /dev/fd0 /mnt/floppy

# вместо

#   sudo usermount /dev/fd0 /mnt/floppy

#  Подобную технику я использую во всех моих
#+ sudo-сценариях, поскольку она кажется мне достаточно удобной.
# -----------------------------------------------------------------

#  Если переменная SUDO_COMMAND не определена, значит сценарий запущен не через
#+ sudo, поэтому повторно вызываем сценарий.
#+ Передвая user id и group id через переменные...

if [ -z "$SUDO_COMMAND" ]
then
   mntusr=$(id -u) grpusr=$(id -g) sudo $0 $*
   exit 0
fi

# В эту точку мы попадаем только если сценарий запущен через sudo
/bin/mount $* -o uid=$mntusr,gid=$grpusr

exit 0

# Дополнительные замечания от автора сценария:
# -------------------------------------------------

# 1) Linux допускает указание опции "users" в файле /etc/fstab,
#    которая позволяет монтировать носители любому пользователю.
#    Но на сервере я предпочитаю дать это право лишь отдельным
#    пользователям. На мой взгляд sudo делает ситуацию более управляемой.

# 2) Я так же считаю, что утилита sudo более удобна, чем
#    выполнение той же задачи посредством групп.

# 3) Эта методика выдает права root на доступ к команде
#    mount, что требует особого внимания при выделении пользователей
#    наделенных таким правом.  Используя ту же самую технику,
#    вы можете более точно разграничить права монтирования
#    устройств, написав сценарии mntfloppy, mntcdrom и mntsamba.
Caution

Слишком глубокая рекурсия может привести к исчерпанию пространства, выделенного под стек, и "вываливанию" сценария по "segfault".

Назад | Вперед
Содержание (общее) | Содержание раздела | Содержание подраздела



Если Вы не нашли что искали, то рекомендую воспользоваться поиском по сайту: