Полное руководство. С# 4.0 - Шилдт Герберт
Шрифт:
Интервал:
Закладка:
Вот к какому результату приводит выполнение этой программы.4/2 равно 2Делить на нуль нельзя!16/4 равно 432/4 равно 8Делить на нуль нельзя!128 / 8 равно 16Подходящий элемент не найден.Подходящий элемент не найден.
Как следует из приведенного выше результата, каждый оператор catch реагируеттолько на свой тип исключения.
Вообще говоря, операторы catch выполняются по порядку их следования в программе. Но при этом выполняется только один блок catch, в котором тип исключения совпадает с типом генерируемого исключения. А все остальные блоки catchпропускаются.Перехват всех исключений
Время от времени возникает потребность в перехвате всех исключений независимоот их типа. Для этой цели служит оператор catch, в котором тип и переменная исключения не указываются. Ниже приведена общая форма такого оператора.catch { // обработка исключений}
С помощью такой формы создается "универсальный" обработчик всех исключений, перехватываемых в программе.
Ниже приведен пример такого "универсального" обработчика исключений. Обратите внимание на то, что он перехватывает и обрабатывает оба исключения,IndexOutOfRangeException и DivideByZeroException, генерируемых в программе.// Использовать "универсальный" обработчик исключений.using System;class ExcDemo5 { static void Main() { // Здесь массив numer длиннее массива denom. int[] numer = { 4, 8, 16, 32, 64, 128, 256, 512 }; int[] denom = { 2, 0, 4, 4, 0, 8 ); for(int i=0; i < numer.Length; i++) { try { Console.WriteLine(numer[i] + " / " + denom[i] + " равно " + numer[i]/denom[i]); } catch { // "Универсальный" перехват. Console.WriteLine("Возникла некоторая исключительная ситуация."); } } }}
При выполнении этой программы получается следующий результат.4/2 равно 2Возникла некоторая исключительная ситуация.16/4 равно 432/4 равно 8Возникла некоторая исключительная ситуация.128 / 8 равно 16Возникла некоторая исключительная ситуация.Возникла некоторая исключительная ситуация.
Применяя "универсальный" перехват, следует иметь в виду, что его блок долженрасполагаться последним по порядку среди всех блоков catch.
ПРИМЕЧАНИЕВ подавляющем большинстве случаев "универсальный" обработчик исключений (не применяется. Как правило, исключения, которые могут быть сгенерированы в коде, обрабатываются по отдельности. Неправильное использование “универсального” обработчика можетпривести к тому, что ошибки, перехватывавшиеся при тестировании программы, маскируются. Кроме того, организовать надлежащую обработку всех исключительных ситуаций в одномобработчике не так-то просто. Иными словами, “универсальный" обработчик исключенийможет оказаться пригодным лишь в особых случаях, например в инструментальном средствеанализа кода во время выполнения.Вложение блоков try
Один блок try может быть вложен в другой. Исключение, генерируемое во внутреннем блоке try и не перехваченное в соответствующем блоке catch, передается вовнешний блок try. В качестве примера ниже приведена программа, в которой исключение IndexOutOfRangeException перехватывается не во внутреннем, а во внешнемблоке try.// Использовать вложенный блок try.using System;class NestTrys { static void Main() { // Здесь массив numer длиннее массива denom. int[] numer = { 4, 8, 16, 32, 64, 128, 256, 512 }; int[] denom = ( 2, 0, 4, 4, 0, 8 ); try { // внешний блок try for(int i=0; i < numer.Length; i++) { try { // вложенный блок try Console.WriteLine(numer[i] + " / " + denom[i] + " равно " + numer[i]/denom[i]); } catch (DivideByZeroException) { Console.WriteLine("Делить на нуль нельзя!"); } } } catch (IndexOutOfRangeException) { Console.WriteLine("Подходящий элемент не найден."); Console.WriteLine("Неисправимая ошибка - программа прервана."); } }}
Выполнение этой программы приводит к следующему результату.4/2 равно 2Делить на нуль нельзя!16/4 равно 432/4 равно 8Делить на нуль нельзя!128 / 8 равно 16Подходящий элемент не найден.Неисправимая ошибка - программа прервана.
В данном примере исключение, обрабатываемое во внутреннем блоке try и связанное с ошибкой из-за деления на нуль, не мешает дальнейшему выполнению программы. Но ошибка нарушения границ массива, обнаруживаемая во внешнем блоке try,приводит к прерыванию программы.
Безусловно, приведенный выше пример демонстрирует далеко не единственноеоснование для применения вложенных блоков try, тем не менее из него можно сделатьважный общий вывод. Вложенные блоки try нередко применяются для обработки различных категорий ошибок разными способами. В частности, одни ошибки считаютсянеисправимыми и не подлежат исправлению, а другие ошибки незначительны и могутбыть обработаны немедленно. Как правило, внешний блок try служит для обнаружениям обработки самых серьезных ошибок, а во внутренних блоках try обрабатываютсяменее серьезные ошибки. Кроме того, внешний блок try может стать "универсальным"для тех ошибок, которые не подлежат обработке во внутреннем блоке.Генерирование исключений вручную
В приведенных выше примерах перехватывались исключения, генерировавшиесяисполняющей системой автоматически. Но исключение может быть сгенерированои вручную с помощью оператора throw. Ниже приведена общая форма такого генерирования:throw exceptOb;
где в качестве exceptOb должен быть обозначен объект класса исключений, производного от класса Exception.
Ниже приведен пример программы, в которой демонстрируется применение оператора throw для генерирования исключения DivideByZeroException.// Сгенерировать исключение вручную.using System;class ThrowDemo { static void Main() { try { Console.WriteLine("До генерирования исключения."); throw new DivideByZeroException(); } catch (DivideByZeroException) { Console.WriteLine("Исключение перехвачено."); } Console.WriteLine("После пары операторов try/catch."); }}
Вот к какому результату приводит выполнение этой программы.До генерирования исключения.Исключение перехвачено.После пары операторов try/catch.
Обратите внимание на то, что исключение DivideByZeroException было сгенерировано с использованием ключевого слова new в операторе throw. Не следуетзабывать, что в данном случае генерируется конкретный объект, а следовательно, ондолжен быть создан перед генерированием исключения. Это означает, что сгенерировать исключение только по его типу нельзя. В данном примере для создания объектаDivideByZeroException был автоматически вызван конструктор, используемый поумолчанию, хотя для генерирования исключений доступны и другие конструкторы.Повторное генерирование исключений
Исключение, перехваченное в одном блоке catch, может быть повторно сгенерировано в другом блоке, чтобы быть перехваченным во внешнем блоке catch. Наиболеевероятной причиной для повторного генерирования исключения служит предоставление доступа к исключению нескольким обработчикам. Допустим, что один обработчикоперирует каким-нибудь одним аспектом исключения, а другой обработчик — другимего аспектом. Для повторного генерирования исключения достаточно указать оператор throw без сопутствующего выражения, как в приведенной ниже форме.throw ;
Не следует, однако, забывать, что когда исключение генерируется повторно, то ононе перехватывается снова тем же самым блоком catch, а передается во внешний блокcatch.
В приведенном ниже примере программы демонстрируется повторное генерирование исключения. В данном случае генерируется исключениеIndexOutOfRangeException.// Сгенерировать исключение повторно.using System;class Rethrow { public static void GenException() { // Здесь массив numer длиннее массива denom. int[] numer = { 4, 8, 16, 32, 64, 128, 256, 512 }; int[] denom = { 2, 0, 4, 4, 0, 8 }; for(int i=0; i<numer.Length; i++) { try { Console.WriteLine(numer[i] + " / " + denom[i] + " равно " + numer[i]/denom[i]); } catch (DivideByZeroException) { Console.WriteLine("Делить на нуль нельзя!"); } catch (IndexOutOfRangeException) { Console.WriteLine("Подходящий элемент не найден."); throw; // сгенерировать исключение повторно } } }}class RethrowDemo { static void Main() { try { Rethrow.GenException(); } catch(IndexOutOfRangeException) { // перехватить исключение повторно Console.WriteLine("Неисправимая ошибка - программа прервана."); } }}