Добавил:
Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:

Секреты программирования для Internet на Java

.pdf
Скачиваний:
181
Добавлен:
02.05.2014
Размер:
3.59 Mб
Скачать

Вычерчивая диаграмму очередного класса или интерфейса, имейте это в виду.

Превращение проекта в работающий код

И вот наступает момент, когда уже пора переходить от диаграмм к составлению реальной программы. Как мы уже говорили в главе 2, пакет представляет собой контейнер, в который помещаются классы, интерфейсы и исключения. Для того чтобы обозначить, что объект относится к определенному пакету, используется ключевое слово package вместе с присвоенным пакету именем:

package somePackage; public class someClass { // определение класса

}

Если исходный код размещается где-то в иерархии каталогов, в качестве разделителей имен этих каталогов используются точки:

package myPackages.someOtherPackage; public class someOtherClass {

// определение класса

}

В любом случае название пакета должно строго соответствовать имени каталога, присвоенного переменной окружения CLASSPATH, о которой мы говорили в главе 2. Точки в названии пакета разделяют имена подкаталогов каталога в переменной CLASSPATH. Для чего предназначены пакеты? Одна из причин - синтаксическая. Компилятору Java нужен способ находить исходный текст программы, а вы не хотите, чтобы все программы находились в одном и том же каталоге. Другая, самая важная причина состоит в наличии модификатора protected. До сих пор мы использовали его не слишком часто. Взамен мы определяли все методы и переменные, к которым можно было обращаться только из объекта, с модификатором private, а все остальные компоненты с модификатором public. Модификатор protected также делает компонент публичным, но в пределах того пакета, которому он принадлежит.

Зачем нам это нужно? Иногда требуется, чтобы определенный метод был доступен лишь некоторым объектам, но не всем. Давайте рассмотрим вышеприведенный пример по управлению воздушными сообщениями. Нашему самолету необходим метод, позволяющий найти свободную ВПП для посадки, но информация к нему может поступить только из диспетчерской. Метод setRunway, принадлежащий классу самолетов, объявляется с модификатором protected и помещается в тот же пакет, в котором описана авиационная диспетчерская. В этот пакет нельзя помещать некоторые классы, например класс "взлетно-посадочная полоса". Представляете, что случится, если этот класс по своему усмотрению вдруг обратится к методу setRunway с разрешением садиться на занятую полосу!

Вы можете сказать: "Все эти классы отнюдь не живут собственной жизнью. Ведь их запрограммировал я, и кому как не мне знать, что от класса "взлетно-посадочная полоса" не может последовать обращения к методу setRunway класса "самолет". Да, в случае простого проекта сказанное вами будет правдой. Но когда дело доходит до крупных проектов, некоторые из модулей действительно начинают жить в определенном смысле собственной жизнью. Как только закончена первичная реализация проекта, кто-нибудь может прийти и попытаться сделать что-то вступающее в полное противоречие с фундаментальными принципами, по которым система проектировалась. При этом совершенно не имеет значения, хорошо ли система была документирована. Лучшая документация в этом случае - это просто запретить компилятору обращаться к методу из класса, из которого таких обращений не было предусмотрено. Модификатор protected имеет смысл применять даже в сравнительно маленьких проектах, над которыми будете работать только вы. Программистам свойственно забывать через некоторое время о том, как была спроектирована программа, и, обратившись к собственному тексту через пару месяцев, вы на самом деле превращаетесь в того "другого" программиста, способного внести полный хаос в отлично задуманную и замечательно реализованную программу.

Те же рассуждения применимы и к модификатору final. Он не относится к концепции пакетов, однако с тем же успехом препятствует ненужной самодеятельности других людей, сопровождающих вашу программу. Как было рассмотрено в главе 4, модификатор final обозначает, что класс, метод или переменная не могут иметь подклассов. С одной стороны, это препятствует конструированию универсальных компонентов, но, с другой стороны, иногда нам вовсе не хочется, чтобы другие люди получили возможность поменять класс, метод или

Ⱦɚɧɧɚɹ ɜɟɪɫɢɹ ɤɧɢɝɢ ɜɵɩɭɳɟɧɚ ɷɥɟɤɬɪɨɧɧɵɦ ɢɡɞɚɬɟɥɶɫɬɜɨɦ %RRNV VKRS Ɋɚɫɩɪɨɫɬɪɚɧɟɧɢɟ ɩɪɨɞɚɠɚ ɩɟɪɟɡɚɩɢɫɶ ɞɚɧɧɨɣ ɤɧɢɝɢ ɢɥɢ ɟɟ ɱɚɫɬɟɣ ɁȺɉɊȿɓȿɇɕ Ɉ ɜɫɟɯ ɧɚɪɭɲɟɧɢɹɯ ɩɪɨɫɶɛɚ ɫɨɨɛɳɚɬɶ ɩɨ ɚɞɪɟɫɭ piracy@books-shop.com

переменную путем создания подклассов на основе нашего "конечного" класса. Давайте рассмотрим следующий пример, касающийся класса "взлетно-посадочная полоса":

class Runway {

private static final phisicalLocation; private static final phisicalLength; private static final phisicalWidth;

//другие переменные

Runway(int i, int j, int k) { physicalLocation = i; physicalLength = j; physicalWidth = k;

}

//другие конструкторы

public final int getPhysicalLocation() { return physicalLocation;

}

public final int getPhysicalLength() { return physicalLength;

}

public final int getPhysicalWidth() { return physicalWidth;

}

// другие методы

}

В отличие от физических размеров взлетно-посадочной полосы, которые не меняются во времени, длина пробега самолета сильно зависит от состояния погодных условий. Таким образом, физические размеры ВПП (переменные и методы, имеющие к ним отношение) объявляются с модификатором final и не подлежат изменениям со стороны других методов.

Как и модификатор protected, модификатор final позволяет защитить разработку программы, воплощенную в готовом коде, от внешних некорректных на нее посягательств. Применение этих модификаторов облегчает дальнейшее сопровождение программы, а также делает ее компонеты более универсальными.

Техника приведения типов объектов

Одним из способов повторного использования класса является создание его подкласса. При этом необходимо дописать лишь несколько отсутствующих у класса-родителя методов и получить класс, полностью готовый к работе. Недостающие методы потомка уже описаны в классеродителе. Проектируя большую систему, необходимо учитывать преимущества Java, относящиеся к приведению типов объектов.

Давайте еще раз обратимся к классу "самолет". Когда мы говорили о способах обобщения модулей - составляющих программы, мы предложили сделать этот класс подклассом более общего типа "транспортное средство" (vehicle). Класс "транспортное средство" должен обладать следующими характеристиками:

abstract class Vehicle { String vehicleType(); Vector getLocation(); Vector getTrajectory(); Path getPath();

float currentFuelAmount();

// возвращает значение от 0 до 1

}

Метод getPath возвращает абстрактный объект Path:

abstract class Path { String pathType();

Vector startingPoint(); Vector endingPoint(); int Length();

www.books-shop.com

}

Все, что нами было написано до этого момента, представляет скорее лишь академический интерес. Разумеется, ваши коллеги высоко оценят объектно-ориентированный стиль программирования, однако нам необходимо сделать что-то практически полезное. Преимущества станут очевидными, как только мы создадим подклассы наших абстрактных классов для разных типов транспортных средств (Vehicle) и их маршрутов (Path). Далее, экземплярами этих подклассов можно будет управлять при помощи универсальных методов, описанных в классеродителе "транспортное средство". То же самое можно сказать и об интерфейсах, подклассы которых образуются аналогичным образом.

Проверка кода на устойчивость

Программы необходимо снабжать средствами защиты от разнообразных ошибок, увы, иногда случающихся при исполнении. Пользователи не всегда вводят правильные данные, а программа не всегда получает все требуемые для нормальной работы ресурсы от системы. В более древних языках программирования защита от ошибок реализовывалась в виде многочисленных и трудночитаемых блоков вида if-then-else. В языке Java используется механизм обработки исключений (исключительных ситуаций) - он здорово помогает, эффективно защищая программу от возникновения ошибочных ситуаций, однако не является магической панацеей от всех бед. Вам все равно придется самостоятельно выяснять места в программе, способные привести к ошибке, и писать соответствующий код для их обработки. С другой стороны, обработка исключений делает исходный текст проще, очевиднее и удобнее в сопровождении.

СОВЕТ Механизм обработки исключений в Java эквивалентен тому же механизму в C++. Однако к нему добавлен блок finally, выполняющийся после любого блока catch.

Нам уже неоднократно приходилось пользоваться механизмом обработки исключений. Многие методы и конструкторы из интерфейса прикладного программирования (API) генерируют исключения. Исключение генерируется в том случае, если внутри данного метода происходит что-то неожиданное, а блок catch решает, что с этой неожиданностью делать. Поскольку нам уже приходилось иметь дело с исключениями, начнем сразу с того, как сделать эффективнее их обработку. Когда вы узнаете, как пишутся методы и конструкторы, генерирующие исключения, вы научитесь писать их самостоятельно. Завершим мы наш экскурс написанием собственных подклассов-исключений, позволяющих максимально приблизить обработку ошибочных ситуаций к конкретным требованиям проекта.

Перехват исключений

Впервые с обработкой исключений вы встретились в главе 5, в процессе загрузки по сети графических файлов. Наверное, вы помните, что конструктор URL генерирует исключение, если не может преобразовать переменную-аргумент типа String в формат URL. Давайте посмотрим, что тут можно сделать:

public class someApplet extends Applet { Image I;

// другие описания public void init() {

String imageName = getParam("image"); try {

URL imageURL = new URL(imageName);

//он может сгенерировать исключение MalformedURLException I = getImage(imageURL);

//инициализируем остальные части апплета

}

catch (MalformedURLException e) { String err = e.getMessage();

showStatus("Couldn't get - +imageName + ":" + e.getMessage); stop();

}

www.books-shop.com

// другие методы

}

Вот простой пример, демонстрирующий использование исключений. Блок try определяет то, что мы намереваемся делать, если все в порядке, тогда как блок catch - то, что необходимо сделать, если в блоке try появится исключение MalformedURLException.

Обработка исключений - важная составляющая правил корректного обращения с любым методом или конструктором программы. Но что такое корректная обработка исключения? Синтаксис нашего примера настолько прост, что не дает полноценного ответа на этот вопрос. Мы должны познакомиться с возможностями обработки исключений в полном масштабе. А именно, необходимо рассмотреть следующие вопросы:

Исключения являются полноправными классами языка Java. Раз так, то в них существуют методы, вызывая которые можно извлечь более подробную информацию о происходящих в системе событиях.

Некоторые исключения должны обрабатываться явным образом, тогда как другие - нет. Первые называются "исключениями времени выполнения - (RuntimeException) и должны обрабатываться в любом случае.

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

Совместно с блоками try и catch можно применять блоки finally.

Изучая тонкости методов обработки исключений, мы будем продолжать разработку нашего простого примера. К концу исследования у вас сформируется четкое представление о том, как правильно получить доступ к ресурсам, основываясь на информации из принимаемой HTMLстраницы. Кроме того, вы научитесь применять концептуальные средства, предназначенные для повышения эффективности использования методов и конструкторов, генерирующих исключения.

Исключения в качестве объектов

Наш блок catch включал следующий оператор: showStatus(e.getMessage());

Здесь подразумевается, что e - это объект, один из членов которого, метод getMessage, возвращает значение типа String. Разумеется, все эти предположения абсолютно правильны. Все исключения, которые можно захватить, являются подклассами класса Throwable, находящегося в пакете java.lang. Поскольку Throwable всегда является суперклассом всякого захватываемого исключения, для анализа ситуации можно пользоваться любым из его методов, определенных с модификатором private. Эти методы перечислены в табл. 10-2.

Таблица 10-2. Методы класса Throwable, определенные с модификатором private

Метод

Описание

String getMessage()

Возвращает детализированное сообщение. Если объект Thowable

 

сконструирован при помощи строки String,, она выводится на экран.

void

Печатает содержимое стека трассировки. Содержимое стека - это все

printStackTrace()

методы,, прерванные в момент генерации исключения. String toString(),

 

Возвращает описание.

Throwable

Заполняет содержимое стека трассировки. Полезен в случае,, если

fillInStackTrace()

обрабатываемое исключение генерируется повторно.

Методы класса Throwable могут быть переопределены подклассом Exception, и к ним, разумеется, могут быть добавлены дополнительные методы. В документации API исключения описаны на соответствующих Web-страницах. Пример такой страницы приведен на рис. 10-4.

www.books-shop.com

Рис. 10.4.

СОВЕТ Не стоит беспокоиться о том, как поступать с различными объектами, передаваемыми при генерации исключений. Ни метод, ни коструктор просто не имеют права сгенерировать объектисключение, например String. Дело в том, что String не является подклассом класса Throwable. Единственный тип объекта, который допустимо передавать при генерации исключения, - это тип, являющийся подклассом Throwable.

Различные типы исключений

До настоящего момента мы относились к исключениям как к данностям, которые можно захватывать с помощью операторов, входящих в блоки catch. Теперь вы знаете, что исключения входят в общую иерархию, основанием которой является класс Throwable. Кроме класса Exception, от основания Throwable отходят еще две ветви специального назначения: класс Error и класс RuntimeException. Три этих класса и отношения между ними показаны на рис. 10-5.

www.books-shop.com

Рис. 10.5.

Три группы, объединенные общим родителем, классом Throwable, отличаются друг от друга функционально и стилистически. Нам как программистам важны скорее их функциональные особенности. Члены второй группы, называемые просто исключениями (Exception), должны обязательно обрабатываться в тех случаях, когда их генерация возможна. Это значит, что данный фрагмент кода не будет компилироваться из-за того, что исключение

MalformedURLException не обрабатывается:

URL U = new URL("http://www.vmedia.com/?.?"); Image I = getImage(U);

//компилироваться не будет - нет блоков try-catch!

В отличие от просто исключений, ошибки (Error) и исключения времени выполнения (Runtime Exception) не нужно обрабатывать явным образом. Если бы мы были обязаны делать это, исходный текст программы превратился бы в непроходимые дебри. Например, если бы мы были должны захватывать исключения типа OutOfMemoryError ("недостаточно памяти"), это приходилось бы делать после каждого оператора, генерирующего любой новый объект! То же самое относится и к исключению NullPointerException ("нулевой указатель"), генерируемому при попытке доступа к нестатическому члену какого-либо объекта до того, как он был реализован. Следующий фрагмент кода демонстрирует один из способов генерации этого исключения:

String S; S.toLowerCase();

// будет сгенерировано исключение NullPointerException

Если бы мы были обязаны обрабатывать исключение NullPointerException, это приходилось бы делать каждый раз при обращении к любому объекту. Это было бы не менее болезненно, чем обязательная обработка OutOfMemoryError.

Основное различие между исключениями времени выполнения и ошибками состоит в типах возникающих в системе сбоев, которые они представляют. Ошибки представляют собой ситуации, в которых ваша программа вряд ли может чем-нибудь помочь. Например, вряд ли программа сможет справиться с ошибкой типа "недостаточно памяти". Поэтому в обработке ошибок программой немного смысла. Лучшее, что можно сделать в этой ситуации, - корректно завершить работу приложения.

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

String S="20"; try {

int i = Integer.parseInt(S);

}

www.books-shop.com

catch (NumberFormatException e) { System.out.println("Couldn't convert - + S);

}

Мы не обязаны обрабатывать исключение времени выполнения NumberFormatException. Но если значение строки S появляется в результате пользовательского ввода, обработать его совершенно необходимо. Если пользователь ошибется, а возникнувшее при этом исключение не будет обработано, то программа, не подозревая ничего дурного, продолжит работу, что приведет к ее аварийному останову через некоторое время.

Обработка суперкласса класса Exception

До настоящего момента нам приходилось обрабатывать именно те исключения, которые могли быть сгенерированы в определенном месте программы. Но поскольку все исключения подчинены общим правилам наследования классов в Java, обрабатывать можно не только отдельные исключения, но также их суперклассы - вплоть до основного класса Object. Мы можем обработать сам класс Throwable - в этом случае в ловушку попадет все, что только может генерироваться. (К сожалению, в этом случае у нас не будет возможности понять, что именно произошло в системе.) С другой стороны, если мы имеем дело с весьма расширенной иерархией исключений, удобнее было бы установить обработку одного суперкласса из этой иерархии, нежели обрабатывать каждое исключение в отдельности.

Несколько блоков catch в одной конструкции

Итак, система обработки исключений заставляет нас внимательно относиться к возможным неожиданностям в работе приложения. К сожалению, на примере 10-1 нам не удалось толком продемонстрировать, чем же обработка исключений Java лучше обыкновенной конструкции if- then-else. В нем на один блок try приходился один блок catch. На самом деле блоков catch, ассоциированных с блоком try, может быть сколько угодно. Это значит, что операторы блока try исполняются при нормальном ходе выполнения программы.

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

Давайте рассмотрим пример, демонстрирующий эти достоинства. Раньше мы просто загружали единственный графический файл в апплет. Теперь усложним задачу: предположим, что Webдизайнер хочет загружать несколько файлов, указывая время, в течение которого каждый из них должен отображаться на экране, - в общем, создать анимацию. Если один из указанных файлов недоступен в момент выполнения апплета, он заменяется на специальное изображение, говорящее дизайнеру об отсутствии данного файла на сервере, так же как специальный значок на HTML-странице говорит об отсутствии соответствующего изображения автору страницы. Если время отображения не указано, используется время по умолчанию. Нашу анимационную задачу решает следующее приложение:

class anumApplet extends Applet { Image imgs[];

Integer pauses[]; // другие описания public void init() {

Vector imgVec = new Vector(); Vector pauseVec = new Vector(); Image deadImage = createDeadImage();

Integer defaultPause = new Integer(100); int indx = 1;

String imgName = getParam("image"+indx); while (imgName != null) {

try {

URL imgURL = new URL (getDocumentBase, imgName);

// здесь может возникнуть MalformedURLException imgVec.add(getImage(imgURL));

String pauseParam = getParam ("pause" + indx); Integer thisPause(pauseParam);

// здесь может возникнуть NumberFormatException pauseVec.add(thisPause);

}

www.books-shop.com

catch (MalformedURLException e) { showStatus(e.getMessage()); System.out.println(e.getMessage()); imgVec.add(deadImage);

}

catch (NumberFormatException e) { pauseVec.add(defaultPause);

}

finally {indx = indx + 1;}

}

imgs = new Image[imgVec.size()]; imgVec.copyInto(imgs); pauseVec.copyInto(pauses);

}

До тех пор пока все идет нормально, выполняются все операторы блока try и в заключение - операторы блока finally. Если другой человек заглянет в этот текст, он без труда поймет, что происходит в программе, а заглянув в блоки catch, выяснит, как вы намереваетесь поступить с каждой из возможных ошибок.

Давайте помотрим, насколько затруднилось бы чтение исходного текста в случае, если бы в Java не было механизма обработки исключений. Сейчас мы рассмотрим небольшой текст, являющийся псевдокодом, а не настоящей программой; поскольку некоторые исключения должны обрабатываться в обязательном порядке, следующий фрагмент компилироваться не будет. Для целей нашего исследования мы придумали несколько не существующих на самом деле методов, без которых было бы не обойтись, если бы в Java отсутствовал механизм обработки исключений. Во-первых, это метод validURL (возвращает boolean), равный true, если URL корректен. Во-вторых, это validIntString, возвращающий true, если String представляет собой целое число:

//Java-псевдокод

//Скомпилировать его невозможно! public void init() {

Vector imgVec = new Vector(); Vector pauseVec = new Vector(); Image deadImage = createDeadImage();

Integer defaultPause = new Integer(100); int indx = 1;

String imgName = getParam("image"+indx); while (imgName != null) {

if (validURL(imgName)) {

URL imgURL = new URL(imgName); imgVec.add(getImage(imgURL)); String pauseParam("pause"+indx); if (validIntString(pauseParam)) {

Integer thisPause(pauseParam); pauseVec.add(thisPause);

}

else pauseVec.add(defaultPause);

else {

showStatus(imgName+" invalid"); System.out.println(imgName+" invalid");

}

indx ++;}

}

Обратите внимание, как сложно разобраться в происходящем в программе, когда не случается никаких ошибок. Кроме того, здесь нет никакого блока try, куда можно было бы заглянуть в случае необходимости. К тому же фрагмент текста программы, реализующий те же функции, что и рассмотренный ранее, вырос в размерах.

СОВЕТ Мы уже говорили о том, что существует возможность обрабатывать суперкласс исключений. Если вы попытаетесь поставить два блока catch, один из которых захватывает суперкласс, а другой - собственно исключение, компилятор заявит, что второй блок catch

www.books-shop.com

недостижим (unreachable) в процессе выполнения программы, если только он не захватывает исключение, генерируемое оболочкой времени выполнения.

Генерация исключений

Мы знаем, как пользоваться методами, генерирующими исключения. Теперь можно приступить к написанию собственных методов этого типа. Используемый при этом синтаксис вполне очевиден. Давайте рассмотрим следующий пример:

class sampleConvert {

public static byte intToByte(int I) throws Ecxeption { if (I > 127 || I < -128) throw Exception();

//здесь не нужен оператор else

//управление автоматически передается блоку catch return (byte)I;

}

//другие методы

}

Если мы генерируем исключение, не относящееся к ошибкам или исключениям времени выполнения, его нужно объявить в соответствующем разделе метода. Вскоре мы увидим, что метод способен генерировать более одного исключения. Для того чтобы показать, что метод генерирует исключение, мы используем ключевое слово throw. Что если метод, вызвавший наш метод, не способен обработать исключение самостоятельно? Он может перехватить его и отправить дальше, к методу, из которого сам был вызван. После оператора throw обязательно должен идти экземпляр подкласса Throwable. Разумеется, мы можем использовать любой доступный для нас конструктор.

Сейчас мы приступим к разработке класса, в процессе которой получим хороший урок на тему "как обрабатывать исключения". В главе 4 было показано, что численные примитивы всегда могут приводиться друг к другу. Даже если мы преобразуем тип int к типу byte и выходим за границы диапазона возможных значений типа byte, преобразование не приведет к ошибке - мы просто получим бессмысленный результат. Но что если нам нужно знать, произошла ли ошибка в процессе преобразования или нет? Лучший способ достичь этого - создать специальный класс Convert, содержимым которого являлись бы методы, предназначенные для проверки такой ситуации. Если преобразование типов приводит к потере информации, метод бы генерировал исключение. Для каждого вида преобразования мы разработаем соответствующий метод, который будет генерировать исключение, как только преобразование типов будет приводить к потере информации. Вот один из таких методов:

public static byte toByte(int I) throws Exception { String mesg = "toByte: ";

if (I > 127) {

mesg = mesg + I + - is less than byte range"; throw new Exception(mesg);}

if (I > 128) {

mesg = mesg + I + - is less than byte range"; throw new Exception(mesg);}

return (byte)I;}

С одной стороны, наш метод - неплохой образец того, как надо обрабатывать исключения. Если бы мы не могли генерировать исключения, у нас не было бы способа сообщить о том, что значение interger вышло за допустимый диапазон.

Однако наш простейший пример абсолютно непригоден для практического применения. Если что-то идет не так, как хотелось бы, наши программы должны знать, почему. Иначе люди, которые, возможно, будут изучать ваш код в дальнейшем, не поймут, что за ошибка произошла в программе. Кроме того, как вы помните, на каждый тип исключения должен приходиться единственный оператор catch. Это значит, что кто-либо, использующий ваш метод, может оказаться не в состоянии отличить один экземпляр исключения от экземпляра исключения другого метода.

Это проблема семантического характера, поскольку в экземпляре класса Exception немного смысла. Нам нужно написать собственные подклассы класса Exception и снабдить их значимыми именами. Так как каждый метод в нашем классе-преобразователе должен генерировать свое собственное исключение, мы разработаем иерархию исключений, как показано на рис. 10-6.

www.books-shop.com

Рис. 10.6.

Написать программу, следующую этой иерархии, очень просто. Все, что необходимо сделать, - это добавить конструктор для различных типов исключений, входящих в части иерархий MagnitudeLoss и PrecisionLoss. Помните, что, как и с любыми классами, каждое исключение должно помещаться в отдельный файл:

Пример 10-1a. Исключения преобразования типов. public class ConvertException extends Exception {

public ConvertException(String S) { super("Couldn't convert: "+S);}

}

public class StringFormatException extends ConvertException { public StringFormatException(String S) {

super("Bad String Format: "+S);}

}

public class MagnitudeLossException extends ConvertException { public MagnitudeLossException(String S) {

super("magnitude loss: - + S);}

}

public class MagnitudeLossException(double d) { this(""+d);}

}

public class LongMagnitudeLossException extends MagnitudeLossException { public LongMagnitudeLossException(long l){

super("long "+l);}

}

public class IntMagnitudeLossException extends MagnitudeLossException { public IntMagnitudeLossException(int i) {

super("int "+i);}

}

public class ShortMagnitudeLossException extends MagnitudeLossException { public ShortMagnitudeLossException(short s) {

super("short - + s);}

}

public class PrecisionLossException extends ConvertException { public PrecisionLossException(String S) {

super("Precision loss: "+S);}

}

public class DoublePrecisionLossException extends PrecisionLossException { public DoublePrecisionLossException(double d) {

super("double: - + d);}

}

public class FloatPrecisionLossException extends PrecisionLossException { public FloatPrecisionLossException(float f) {

super("float: - + f);}

}

www.books-shop.com