Язык программирования C#9 и платформа .NET5
Представленная выше логика обработки исключений приводит к возникновению ошибок на этапе компиляции. Проблема в том, что первый блок
способен обрабатывать любые исключения, производные отcatch(с учетом отношения "является"), в том числеSystem.ExceptionиCarIsDeadException. Следовательно, два последних блокаArgumentOutOfRangeExceptionв принципе недостижимы!catchЗапомните эмпирическое правило: блоки
должны быть структурированы так, чтобы первыйcatchперехватывал наиболее специфическое исключение (т.е. производный тип, расположенный ниже всех в цепочке наследования типов исключений), а последнийcatch— самое общее исключение (т.е. базовый класс имеющейся цепочки наследования:catchв данном случае).System.ExceptionТаким образом, если вы хотите определить блок
, который будет обрабатывать любые исключения помимоcatchиCarIsDeadException, то можно было бы написать следующий код:ArgumentOutOfRangeException// Этот код скомпилируется без проблем.Console.WriteLine("***** Handling Multiple Exceptions *****\n");Car myCar = new Car("Rusty", 90);try{// Вызвать исключение выхода за пределы диапазона аргумента.myCar.Accelerate(-10);}catch (CarIsDeadException e){Console.WriteLine(e.Message);}catch (ArgumentOutOfRangeException e){Console.WriteLine(e.Message);}// Этот блок будет перехватывать все остальные исключения.// помимо CarIsDeadException и ArgumentOutOfRangeExceptioncatch (Exception e){Console.WriteLine(e.Message);}Console.ReadLine();На заметку! Везде, где только возможно, отдавайте предпочтение перехвату специфичных классов исключений, а не общего класса
. Хотя может показаться, что это упрощает жизнь в краткосрочной перспективе (поскольку охватывает все исключения, которые пока не беспокоят), в долгосрочной перспективе могут возникать странные аварийные отказы во время выполнения, т.к. в коде не была предусмотрена непосредственная обработка более серьезной ошибки. Не забывайте, что финальный блокSystem.Exception, который работает сcatch, на самом деле имеет тенденцию быть чрезвычайно общим.System.ExceptionОбщие операторы catch
В языке C# также поддерживается "общий" контекст
, который не получает явно объект исключения, сгенерированный заданным членом:catch// Общий оператор catch.Console.WriteLine("***** Handling Multiple Exceptions *****\n");Car myCar = new Car("Rusty", 90);try{myCar.Accelerate(90);}catch{Console.WriteLine("Something bad happened...");// Произошло что-то плохое...}Console.ReadLine();Очевидно, что это не самый информативный способ обработки исключений, поскольку нет никакой возможности для получения содержательных данных о возникшей ошибке (таких как имя метода, стек вызовов или специальное сообщение). Тем не менее, в C# такая конструкция разрешена, потому что она может быть полезной, когда требуется обрабатывать все ошибки в обобщенной манере.
Повторная генерация исключений
Внутри логики блока
перехваченное исключение разрешено повторно сгенерировать для передачи вверх по стеку вызовов предшествующему вызывающему коду. Для этого просто используется ключевое словоtryв блокеthrow. В итоге исключение передается вверх по цепочке вызовов, что может оказаться полезным, если блокcatchспособен обработать текущую ошибку только частично:catch// Передача ответственности....try{// Логика увеличения скорости автомобиля...}catch(CarIsDeadException e){// Выполнить частичную обработку этой ошибки и передать ответственность.throw;}...Имейте в виду, что в данном примере кода конечным получателем исключения
будет исполняющая среда .NET 5, т.к. операторы верхнего уровня генерируют его повторно. По указанной причине конечному пользователю будет отображаться системное диалоговое окно с информацией об ошибке. Обычно вы будете повторно генерировать частично обработанное исключение для передачи вызывающему коду, который имеет возможность обработать входное исключение более элегантным образом.CarIsDeadExceptionТакже обратите внимание на неявную повторную генерацию объекта
с помощью ключевого словаCarIsDeadExceptionбез аргументов. Дело в том, что здесь не создается новый объект исключения, а просто передается исходный объект исключения (со всей исходной информацией). Это позволяет сохранить контекст первоначального целевого объекта.throw