Г. Шилдт - Полный справочник по C# (1160789), страница 67
Текст из файла (страница 67)
Язык С#Стандартное сообщение таково:System.IndexOutOfRangeException: Index was outsidethe bounds of the array.at ExcTest.genException()at UseExcept.Main()Свойство StackTrace:at ExcTest.genException()at UseExcept.Main()Свойство Message: Index was outside the bounds of the array.Свойство TargetSite: Void genException()После catch-инструкции.В классе Exception определено четыре конструктора. Наиболее часто используются такие:Exception()Exception(string str)Первый — это конструктор по умолчанию.
Второй принимает значение свойстваMessage, связанное с исключением. При создании собственных классов исключенийнеобходимо реализовать оба этих конструктора.Наиболее употребительные исключенияВ пространстве имен System определено несколько стандартных встроенных исключений. Все они выведены из класса SystemException, поскольку генерируютсясистемой динамического управления (Common Language Runtime) при возникновениидинамических ошибок. Некоторые из самых употребительных стандартных исключений, определенных в С#, приведены в табл. 13.1.Таблица 13.1. Наиболее употребительные исключения, определенные в пространстве именSystemИсключениеЗначениеArrayTypeMismatchExceptionТип сохраняемого значения несовместим с типом массиваDivideByZeroExceptionПопытка деления на нульIndexOutOfRangeExceptionИндекс массива оказался вне диапазонаinvalidCastExceptionНеверно выполнено динамическое приведение типовOutof MemoryExceptionОбращение к оператору new оказалось неудачным из-за недостаточногообъема свободной памятиoverf lowExceptionИмеет место арифметическое переполнениеNuilRef erenceExceptionБыла сделана попытка использовать нулевую ссылку, т.е.
ссылку, котораяне указывает ни на какой объектПереполнение стекаstackoverf lowExceptionБольшинство исключений, перечисленных в табл. 13.1, не нуждается в дополнительных разъяснениях, за исключением класса NuilRef erenceException. Это исключение генерируется при попытке использовать нулевую ссылку, например, припопытке вызвать метод, передав ему вместо ссылки на объект нулевую ссылку. Нулевая ссылка не указывает ни на какой объект.
Один из способов создать нулевую ссылку — явно присвоить ссылочной переменной null-значение, используя ключевоеслово n u l l . Нулевые ссылки можно получить и другими путями, которые, однако,менее очевидны. Рассмотрим программу, в которой демонстрируется возникновениеисключения типа NullReferenceException.Глава 13. Обработка исключительных ситуаций365// Использование исключения типа NullReferenceException.using System;class X {int x;public X(int a) {x = a;public int add(X o) {return x + o.x;// Демонстрируем исключение типа NullReferenceException.class NREDemo {public static void Main() {X p « new X(10);X q = null; // Переменной q явно присваивается// значение null,int val;try {val = p.add(q); // Такой вызов метода// приведет к исключению.} catch (NullReferenceException) {Console.WriteLine("NullReferenceException!");Console.WriteLine("Исправляем ошибку...\n");// Исправляем ошибку,q = new X(9);val = p.add(q);Console.WriteLine("Значение val равно {О}", val);При выполнении этой программы получаем такие результаты:NullReferenceException!Исправляем ошибку...Значение val равно 19Эта программа создает класс X, в котором определяется член х и метод add (),предназначенный для сложения значения х, принадлежащего вызывающему объекту,с членом х, который определен в объекте, переданном в качестве параметра.
В методеMain () создаются два объекта класса х. Первый, р, инициализируется, а второй, q, —нет (ему явным образом присваивается значение n u l l ) . Затем вызывается методp.iueth (), которому значение q передается как аргумент. Поскольку переменная q нессылается ни на один объект, при попытке получить значение члена q. x генерируетсяисключение типа NullReferenceException.Заслуживает внимания исключение типа StackOverf lowException, которое генерируется при переполнении системного стека. Оно может возникнуть при некорректном определении рекурсивного метода. Программисту, который увлекается рекурсией,366Часть I. Язык С#возможно, стоит внимательно отследить появление исключения этого типа, принявсоответствующие меры в случае его обнаружения.
Однако здесь следует проявить осторожность. Если уж это исключение сгенерировано, значит, системный стек исчерпалсвои возможности, поэтому лучше всего просто начать анализ с рекурсивного вызова.Наследование классов исключенийНесмотря на то что встроенные С#-исключения обрабатывают самые распространенные ошибки, С#-механизм обработки исключений не ограничивается этимиошибками.
В С# имеется возможность обрабатывать исключения, создаваемые программистом. В своих программах вы можете использовать для обработки ошибок"собственные" исключения. В создании исключения нет ничего сложного. Для этогодостаточно определить класс как производный от класса Exception. Как правило,определяемые программистом исключения, должны быть производными от классаApplicationException, "родоначальника" иерархии, зарезервированной для исключений, связанных с прикладными программами. Созданные вами производные классыне должны ничего реализовывать, поскольку одно лишь их существование в системетипов уже позволит использовать их в качестве исключений.Классы исключений, создаваемые программистом, будут автоматически иметьсвойства и методы, определенные в классе Exception и доступные для них.
Конечно,один или несколько членов в новых классах можно переопределить.Рассмотрим пример, в котором создается "пользовательский" тип исключения. Вконце главы 10 был приведен пример разработки класса массива с именемRangeArray. Вспомним, что класс RangeArray поддерживает одномерные i n t массивы, в которых начальный и конечный индексы задаются пользователем. Например, массив, индексы которого лежат в диапазоне от -5 до 27, абсолютно легален дляобъектов класса RangeArray.
В главе 10 было показано, что при попадании индексаза пределы диапазона, устанавливалась переменная ошибки, определенная в классеRangeArray. Это означает, что переменную ошибки необходимо было проверять после каждой операции, в которой участвовал объект класса RangeArray.
Безусловно,такое решение связано с ошибками и лишено "изящества". Предпочтительней, чтобыобъект класса RangeArray при возникновении ошибки нарушения границ диапазонагенерировал "свое" исключение. Именно такое решение и реализовано в следующей^ерсии класса RangeArray.// Создание пользовательского исключения для// обнаружения ошибок при работе с объектами класса// RangeArray.using System;// Создаем исключение для класса RangeArray.c l a s s RangeArrayException : ApplicationException {// Реализуем стандартные конструкторы,public RangeArrayException() : base() { }p u b l i c RangeArrayException(string s t r ) : b a s e ( s t r ){ }// Переопределяем метод ToStringO для класса// RangeArrayException.public override s t r i n g ToStringO {r e t u r n Message;Глава 13.
Обработка исключительных ситуаций367// Улучшенная версия класса RangeArray.class RangeArray {// Закрытые данные.int[] а; // Ссылка на базовый массив.int lowerBound; // Наименьший индекс.int upperBound; // Наибольший индекс.int len; // Базовая переменная для свойства Length.// Создаем массив с заданным размером,public RangeArray(int low, int high) {high++;if(high <= low) {throw new RangeArrayException("Нижний индекс не меньше верхнего."),}а = new int[high - low];len = high - low;lowerBound = low;upperBound = --high;// Свойство Length, предназначенное только для чтения,public int Length {get {return len;// Индексатор для объекта класса RangeArray.public int this[int index] {// Средство для чтения элемента массива,get {if(ok(index)) {return a[index - lowerBound];} else {throw new RangeArrayException("Ошибка нарушения границ диапазона.");// Средство для записи элемента массива.set {if(ok(index)) {a[index - lowerBound] = value;}else throw new RangeArrayException("Ошибка нарушения границ диапазона.");// Метод возвращает значение true,// если индекс в пределах диапазона,private bool ok(int index) {if(index >= lowerBound & index <= upperBound)return true;return false;368Часть I.
Язык С#// Демонстрируем использование массива с заданным// диапазоном изменения индекса,class RangeArrayDemo {public static void Main() {try {RangeArray ra = new RangeArray(-5, 5 ) ;RangeArray ra2 = new RangeArray(1, 10);// Демонстрируем использование объекта-массива га.Console.WriteLine("Длина массива га: " + ra.Length);for(int i = -5; i <= 5; i++)ra[i] = i;Console.Write("Содержимое массива га: " ) ;for(int i = -5; i <= 5; i++)Console.Write(ra[i] + " " ) ;Console.WriteLine("\n");// Демонстрируем использование объекта-массива га2.Console.WriteLine("Длина массива га2: " + ra2.Length);for(int i = 1; i <= 10; i++)ra2[i] = i;Console.Write("Содержимое массива ra2: ") ;for(int i = 1; i <= 10; i++)Console.Write(ra2[i] + " " ) ;Console.WriteLine("\n");} catch (RangeArrayException exc) {Console.WriteLine(exc);}// Теперь демонстрируем "работу над ошибками".Console.WriteLine("Сгенерируем ошибки непопадания в диапазон.");// Используем неверно заданный конструктор,try {RangeArray гаЗ = new RangeArray(100, -10); // Ошибка!} catch (RangeArrayException exc) {Console.WriteLine(exc);}// Используем неверно заданный индекс,try {RangeArray гаЗ = new RangeArray(-2, 2 ) ;for(int i = - 2 ; i <= 2; i++)гаЗ [i] = i;Console.Write("Содержимое массива гаЗ: " ) ;for(int i = -2; i <= 10; i++) // Ошибка непопадания// в диапазон.Console.Write(гаЗ[i] + " " ) ;Глава 13.