Инициализация исключений
Вы можете самостоятельно инициировать исключительную ситуацию при выполнении тех или иных действий. Но, хотя синтаксис конструктора объекта Exception похож на конструкторы всех других объектов, создается он по- особенному. Для этого используется оператор raise, за которым в качестве параметра должен идти экземпляр объекта типа Exception. Обычно сразу за оператором следует конструктор класса ИС: raise EMathError.Create(' '); но можно и разделить создание и возбуждение исключительной ситуации: var E: EMathError; Begin E:= EMathError.Create С'); raise E; end; Оператор raise передает созданную исключительную ситуацию ближайшему блоку try..except (см. ниже). if С = 0 then raise EDivByZero.Create('Деление на ноль') Else А:= В/С; Самостоятельная инициализация ИС может пригодиться при программировании реакции приложения на ввод данных, для контроля значений переменных и т. д. В таких случаях желательно создавать собственные классы ИС, специально приспособленные для ваших нужд. Также полезно использовать специально спроектированные исключительные ситуации при создании собственных объектов и компонентов. Так, многие важнейшие классы VCL — списки, потоки, графические объекты — сигнализируют о своих (или ваших?) проблемах созданием соответствующей ИС — EListError, EInvalidGraphic, EPrinter и т. д. Самый важный отличительный признак объекта Exception — это все же класс, к которому он принадлежит. Именно факт принадлежности возникшей ИС к тому или иному классу говорит о том, что случилось. Если же нужно детализировать проблему, можно присвоить значение свойству Message. Если и этого мало, можно добавить в объект новые поля. Так, в ИС EinOutError (ошибка ввода/вывода) есть поле ErrorCode, значение которого соответствует произошедшей ошибке — запрету записи, отсутствию или повреждению файла и т. д. Try FileOpen('ñ:\myfile.txt', fmOpenWrite); Except On E: EinOutError do Case E.ErrorCode of ERROR_FILE_NOT_FOUND {=2}: ShowMessage('Файл не найден!'); ERROR_ACCESS_DENIED {=5}: ShowMessage('Доступ запрещен!'); ERROR_DISK_FULL {=112}: ShowMessage ('Диск переполнен!'); end; end; Впрочем, ИС EInOutError возникают только тогда, когда установлена опция компилятора {$IOCHECKS ON} (или иначе {$I+}). В противном случае проверку переменной IOResult (известной еще по Turbo Pascal) нужно делать самостоятельно. Защитные конструкции языка Object Pascal Для работы с объектами исключительных ситуаций существуют специальные конструкции языка Object Pascal— блоки try…except и try…finally. Они контролируют выполнение операторов, помещенных внутри блока до ключевого слова except или finally. В случае возникновения исключительной ситуации штатное выполнение вашей программы немедленно прекращается, и управление передается операторам, идущим за указанными ключевыми словами. Если в вашей процедуре эти блоки отсутствуют, управление все равно будет передано ближайшему блоку, внутри которого возникла ситуация. Хотя синтаксис двух видов блоков похож, но они принципиально отличаются назначением и решаемыми задачами. Блок try..except Для реакции на конкретный тип ситуации применяется блок try..except. Синтаксис его следующий: Try <Оператор>; <Оператор>; ... Except On EException1 do <Оператор>; {обработчик ИС типа EException1} On EException2 do <Оператор>; ... Else <0ператор> {обработчик прочих ИС} end; Выполнение блока начинается с секции try. При отсутствии исключительных ситуаций только она и выполняется. Секция except получает управление в случае возникновения ИС. После обработки происходит выход из защищенного блока, и управление обратно в секцию try не передается; выполняются операторы, стоящие после end. Если вы хотите обработать любую ИС одинаково, независимо от ее класса, вы можете писать код прямо между операторами except и end. Но если обработка отличается, здесь можно применять набор директив on…do, определяющих реакцию приложения на определенную ситуацию. Каждая директива связывает ситуацию (on...), заданную своим именем класса, с группой операторов (do...). U:= 220.0; R:= 0; Try I:= U / R; Except on EZeroDivide do MessageBox('Короткое замыкание!'); end; В этом примере замена if…then на try…except, может быть, не дала очевидной экономии кода. Однако если при решении, допустим, вычислительной задачи проверять на возможное деление на ноль приходится не один, а множество раз, то выигрыш от нового подхода неоспорим - достаточно одного блока try…except на все вычисления. При возникновении ИС директивы просматриваются последовательно, в порядке их описания. Каждый тип исключительной ситуации, описанный после ключевого слова on, обрабатывается именно этим блоком: только то, что предусмотрено в нем, и будет являться реакцией на данную ситуацию. Если при этом обработчик родительского класса стоит перед дочерним, последний никогда не получит управления. Try i:=l;j:=0; k:=i div j; Except on EIntError do ShowMessage('IntError'); on EDivByZero do ShowMessage('DivByZero'); end; В этом примере, хотя в действительности будет иметь место деление на ноль (EDivByZero), вы увидите сообщение, соответствующее родительскому классу EIntError. Но стоит поменять две конструкции on…do местами, и все придет в норму. Если возникла ситуация, не определенная ни в одной из директив, выполняются те операторы, которые стоят после ключевого слова else. Если и их нет, то ИС считается не обработанной и будет передана на следующий уровень обработки. Этим следующим уровнем может быть другой оператор try..except, который содержит в себе данный блок.
|