Исключения

Работа с исключениям это отдельное искусство, освоив которое вы получите инструмент огромного потенциала. Я продемонстрирую несколько методов работы с исключениями в этой главе.

В конечном итоге, нас интересует синтаксис try/except. Код, который может вызвать исключение, помещается в try блок, обработка исключения - в except. Простой пример:

try:
    file = open('test.txt', 'rb')
except IOError as e:
    print('Было вызвано исключение IOError. {}'.format(e.args[-1]))

В примере выше мы перехватываем только исключение IOError. Многие новички не знают, что мы можем обрабатывать несколько исключений.

Обработка множества исключений

Мы можем использовать три метода обработки множества исключений. Первый заключается в создании кортежа из всех возможных исключений. Что-то подобное:

try:
    file = open('test.txt', 'rb')
except (IOError, EOFError) as e:
    print("Было вызвано исключение. {}".format(e.args[-1]))

Другой методы заключается в обработке каждого исключения в отдельном блоке except. Мы можем иметь неограниченное их число (но не менее одного). Очередной пример:

try:
    file = open('test.txt', 'rb')
except EOFError as e:
    print("Было вызвано исключение EOFError.")
    raise e
except IOError as e:
    print("Было вызвано исключение IOError.")
    raise e

Таким образом, если исключение не перехватывается первым блоком except, то оно может быть обработано следующим, или не быть обработанным вовсе. Последний метод заключается в перехвате ВСЕХ исключений:

try:
    file = open('test.txt', 'rb')
except Exception:
    # Логирование, если оно вам требуется
    raise

Это может быть полезно, когда вы не знаете какие исключения могут возникнуть в вашей программе.

finally

Основной код помещается в блок try. Дальше идут блоки except, которые исполняются, если в блоке try было вызвано определённое исключение. Третьим типом блоков, следующим за двумя первыми, может быть finally. Код в блоке finally будет исполнен вне зависимости от того, вызвал ли код в блоке try исключение или нет. Это может быть полезно для финальной "чистки" после работы основного скрипта. Вот простой пример:

try:
    file = open('test.txt', 'rb')
except IOError as e:
    print('Было вызвано исключение IOError. {}'.format(e.args[-1]))
finally:
    print("Я буду напечатан вне зависимости от исключений в блоке try!")

# Вывод: Было вызвано исключение IOError. No such file or directory
#        Я буду напечатан вне зависимости от исключений в блоке try!

try/else

Иногда мы можем захотеть исполнить определенный код, если исключения не было. Это легко сделать с помощью блока else. Вы можете спросить: почему нам нужен else, если мы можем поместить этот код в блок try? Проблема в том, что исключение в этом коде, может быть в свою очередь поймано try, а мы можем этого и не хотеть. В целом, else нечасто используется, и я, честно говоря, редко к нему прибегаю сам. Пример:

try:
    print('Я уверен, исключений не будет!')
except Exception:
    print('Исключение')
else:
    # Любой код, который должен быть исполнен, если исключение в блоке
    # try не было вызвано, но для которого не должна проводиться
    # обработка исключений
    print('Я буду исполнен, если в try не будет исключений.'
          'Мои исключения не будут обрабатываться.')
finally:
    print('Я буду исполнен в любом случае!')

# Вывод: Я уверен, исключений не будет!
#        Я буду исполнен, если в try не будет исключений. Мои исключения не будут обрабатываться.
#        Я буду исполнен в любом случае!

Блок else, таким образом, исполняется при отсутствии исключений в блоке try. else исполняется перед finally.

Last updated