Вся полувековая история программирования компьютеров, а может быть, и история всей науки — это попытка совладать со сложностью окружающего мира. Задачи, встающие перед программистами, становятся все более громоздкими, информация, которую надо обработать, растет как снежный ком. Еще недавно обычными единицами измерения информации были килобайты и мегабайты, а сейчас уже говорят только о гигабайтах и терабайтах. Как только программисты предлагают более-менее удовлетворительное решение поставленных задач, тут же возникают новые, еще более сложные задачи. Программисты придумывают новые методики, создают новые языки. За полвека появилось несколько сотен языков, предложено множество методов и стилей программирования. Некоторые методы и стили становятся общепринятыми и образуют на некоторое время так называемую парадигму программирования.
Парадигмы программирования
Первые, даже самые простые программы, написанные в машинных кодах, составляли сотни строк совершенно непонятного текста. Языки ассемблера облегчили чтение программ, но не упростили их. Для упрощения и ускорения программирования придумали языки высокого уровня: FORTRAN, Algol и сотни других, возложив рутинные операции по созданию машинного кода на компилятор. Те же программы, переписанные на языках высокого уровня, стали гораздо понятнее и короче. Но жизнь потребовала решения более сложных задач, и программы снова увеличились в размерах, стали громоздкими и необозримыми.
Возникла идея: оформить программу в виде нескольких по возможности простых процедур или функций, каждая из которых решает свою определенную задачу. Написать, откомпилировать и отладить небольшую процедуру можно легко и быстро. Затем остается только собрать все процедуры в нужном порядке в одну программу. Кроме того, один раз написанные процедуры можно затем использовать в других программах как строительные кирпичики. Процедурное программирование быстро стало парадигмой. Во все языки высокого уровня включили средства написания процедур и функций. Появилось множество библиотек процедур и функций на все случаи жизни.
Встал вопрос о том, как выявить структуру программы, разбить программу на процедуры, какую часть кода выделить в отдельную процедуру, как сделать алгоритм решения задачи простым и наглядным, как удобнее связать процедуры между собой. Опытные программисты предложили свои рекомендации, названные структурным программированием. Структурное программирование оказалось удобным и стало парадигмой. Появились языки программирования, например Pascal, на которых удобно писать структурные программы. Более того, на них очень трудно написать неструктурные программы.
Сложность стоящих перед программистами задач проявилась и тут: программы стали содержать сотни процедур и опять оказались необозримыми. "Кирпичики" стали слишком маленькими. Потребовался новый стиль программирования.
В это же время обнаружилось, что удачная или неудачная структура исходных данных может сильно облегчить или усложнить их обработку. Одни исходные данные удобнее объединить в массив, для других больше подходит структура дерева или стека. Появилось множество исследований различных структур данных и рекомендаций по их применению. Никлаус Вирт, создатель языка Pascal, даже назвал одну из своих книг "Алгоритмы + структуры данных = программы".
Возникла идея объединить исходные данные Рё РІСЃРµ процедуры РёС… обработки РІ РѕРґРёРЅ модуль. Рта идея модульного программирования быстро завоевала СѓРјС‹ Рё РЅР° некоторое время стала парадигмой. Программы составлялись РёР· отдельных модулей, содержащих десяток-РґСЂСѓРіРѕР№ процедур Рё функций. Рффективность таких программ тем выше, чем меньше модули зависят РґСЂСѓРі РѕС‚ РґСЂСѓРіР°. Автономность модулей позволяет создавать Рё библиотеки модулей, чтобы потом использовать РёС… РІ качестве строительных блоков для РґСЂСѓРіРёС… программ.
Для того чтобы обеспечить максимальную независимость модулей друг от друга, надо четко отделить процедуры, которые будут использоваться другими модулями, — открытые (public) процедуры, от вспомогательных, которые обрабатывают данные, заключенные в этот модуль, — закрытых (private) процедур. Для этого модуль делится на две части. Открытые процедуры перечисляются в первой части модуля — интерфейсе (interface), вторые участвуют только во второй его части — реализации (implementation) модуля. Данные, занесенные в модуль, тоже делятся на открытые, указанные в интерфейсе и доступные для других модулей, и закрытые, доступные только для процедур того же модуля. В различных языках программирования это деление производится по-разному. В языке Turbo Pascal модуль специально делится на интерфейс и реализацию, в языке С интерфейс выносится в отдельные "головные" (header) файлы. В языке С++, кроме того, для описания интерфейса можно воспользоваться абстрактными классами. В языке Java есть специальная конструкция для описания интерфейсов, которая так и называется — interface, но можно написать и абстрактные классы.
Так возникла идея о скрытии, инкапсуляции (incapsulation) данных и методов их обработки. Подобные идеи периодически возникают в дизайне бытовой техники. То телевизоры испещряются кнопками и топорщатся ручками и движками на радость любознательному телезрителю, господствует так называемый "приборный" стиль, то вдруг все куда-то пропадает, а на панели управления остаются только кнопка включения телевизора и ручка громкости. Любознательный телезритель, привыкший самостоятельно настраивать свой телевизор, берется за отвертку и лезет внутрь.
Рнкапсуляция, конечно, производится РЅРµ для того, чтобы спрятать РѕС‚ РґСЂСѓРіРѕРіРѕ модуля что-то любопытное. Здесь преследуются РґРІРµ основные цели. Первая — обеспечить безопасность использования модуля, вынести РІ интерфейс, сделать общедоступными только те методы обработки информации, которые РЅРµ РјРѕРіСѓС‚ испортить или уничтожить исходные данные. Вторая цель — уменьшить сложность, скрыв РѕС‚ внешнего РјРёСЂР° ненужные детали реализации.
Опять возник вопрос: каким образом разбить программу на модули? Тут кстати оказались методы решения старой задачи программирования — моделирования действий искусственных и природных объектов: роботов, станков с программным управлением, беспилотных самолетов; моделирования поведения людей, животных, растений, систем обеспечения жизнедеятельности, систем управления технологическими процессами.
Р’ самом деле, каждый объект — СЂРѕР±РѕС‚, автомобиль, человек — обладает определенными характеристиками. РРјРё РјРѕРіСѓС‚ служить: вес, СЂРѕСЃС‚, максимальная скорость, СѓРіРѕР» поворота, грузоподъемность, фамилия, возраст. Объект способен выполнять какие-то действия: перемещаться РІ пространстве, поворачиваться, поднимать РіСЂСѓР·, копать котлован или траншею, расти или уменьшаться, есть, пить, рождаться Рё умирать, изменяя СЃРІРѕРё первоначальные характеристики. РЈРґРѕР±РЅРѕ смоделировать объект РІ РІРёРґРµ модуля. Его характеристики Р±СѓРґСѓС‚ данными, постоянными или переменными, Р° действия — процедурами.
Оказалось удобным сделать и обратное — разбить программу на модули так, чтобы она превратилась в совокупность взаимодействующих объектов. Так возникло объектноориентированное программирование (object-oriented programming), сокращенно ООП (OOP) — современная парадигма программирования.
Р’ начале работы программы создается РѕРґРёРЅ или несколько объектов. Объекты активны. РћРЅРё выполняют СЃРІРѕРё методы обработки информации, сохраняя результаты обработки РІ СЃРІРѕРёС… полях, файлах, базах данных или РІ каких-то РґСЂСѓРіРёС… хранилищах. РџРѕ мере необходимости объекты обращаются Рє методам РґСЂСѓРіРёС… объектов, передавая РёРј нужные сведения. Р’ процессе выполнения программы РјРѕРіСѓС‚ создаваться новые объекты Рё уничтожаться старые, ненужные объекты. Работа программы завершится, РєРѕРіРґР° РѕРґРёРЅ РёР· объектов выполнит метод завершения программы. Ртот метод обычно сохраняет обработанную информацию РІ указанных ему хранилищах данных Рё удаляет РІСЃРµ объекты, освобождая оперативную память Рё РґСЂСѓРіРёРµ ресурсы, занятые объектами.
Р’ РІРёРґРµ объектов можно представить совсем неожиданные понятия. Например, РѕРєРЅРѕ РЅР° экране дисплея — это объект, имеющий ширину width Рё высоту height, определенное расположение РЅР° экране, описываемое обычно координатами (x, Сѓ) левого верхнего угла РѕРєРЅР°, Р° также шрифт, которым РІ РѕРєРЅРѕ выводится текст, скажем, Times New Roman, цвет фона color, несколько РєРЅРѕРїРѕРє, полосы прокрутки Рё РґСЂСѓРіРёРµ характеристики. РћРєРЅРѕ может перемещаться РїРѕ экрану методом, описанным РІ какой-РЅРёР±СѓРґСЊ процедуре, скажем, move (), увеличиваться или уменьшаться РІ размерах каким-РЅРёР±СѓРґСЊ методом size(), сворачиваться РІ ярлык методом iconify(), как-то реагировать РЅР° действия мыши Рё нажатия клавиш. Рто полноценный объект! РљРЅРѕРїРєРё, полосы прокрутки Рё прочие элементы РѕРєРЅР° — это тоже объекты СЃРѕ СЃРІРѕРёРјРё характеристиками Рё действиями: размерами, шрифтами, перемещениями.
Разумеется, считать, что окно само "умеет" выполнять действия, а мы только даем ему поручения: "Свернись, развернись, передвинься", — это несколько неожиданный взгляд на вещи, но ведь сейчас можно подавать команды не только манипуляцией мышью и нажатием клавиш, но и голосом!
Рдея объектно-ориентированного программирования оказалась очень плодотворной Рё стала активно развиваться. Выяснилось, что СѓРґРѕР±РЅРѕ ставить задачу сразу РІ РІРёРґРµ совокупности действующих объектов — РІРѕР·РЅРёРє объектно-ориентированный анализ, РћРћРђ (object-oriented analysis, OOA). Решили проектировать сложные системы РІ РІРёРґРµ объектов — появилось объектно-ориентированное проектирование, РћРћРџ (object-oriented design, OOD).
В начале разработки объектно-ориентированной программы сразу встает множество вопросов. Сколько объектов понадобится для правильной работы программы? Каким образом и в какое время создавать объекты? Как распределить работу между объектами? Как организовать взаимодействие объектов? В объектно-ориентированном программировании выработано несколько общепризнанных принципов, более или менее полно отвечающих на эти вопросы. Познакомимся с ними.
Принципы объектно-ориентированного программирования
Объектно-ориентированное программирование развивается уже несколько десятков лет. Рмеется несколько школ, каждая РёР· которых предлагает СЃРІРѕР№ набор принципов работы СЃ объектами Рё РїРѕ-своему излагает эти принципы. Бурные обсуждения Рё РґРёСЃРєСѓСЃСЃРёРё, проходившие между представителями этих школ, позволили выработать несколько общепринятых принципов, признанных всеми школами Рё внедренных РІРѕ РІСЃРµ объектноориентированные языки программирования. Перечислим эти принципы.
Абстракция
Описывая поведение какого-либо объекта, например автомобиля, мы строим его модель. Модель, как правило, не может описать объект полностью: реальные объекты слишком сложны. Приходится отбирать только те характеристики объекта, которые важны для решения поставленной перед нами задачи. Скажем, для описания грузоперевозок важной характеристикой будет грузоподъемность автомобиля, а для описания автомобильных гонок она не существенна. Но для моделирования гонок обязательно надо описать метод набора скорости данным автомобилем, а для грузоперевозок это не столь важно.
Для характеристики спортсмена обязательно надо указать его вес, рост, скорость реакции, спортивные достижения, а для ученого все эти качества несущественны, зато важны его квалификация, ученая степень, количество опубликованных научных работ.
Мы должны абстрагироваться от некоторых конкретных деталей объекта, отбросить их. Очень важно выбрать правильную степень абстракции. Слишком высокая степень даст только приблизительное описание объекта, не позволит правильно моделировать его поведение. Можно охарактеризовать человека как "Двуногое без перьев", но что это даст для его понимания? С другой стороны, слишком низкая степень абстракции сделает модель очень сложной, перегруженной деталями и потому непригодной.
Например, можно совершенно точно предсказать погоду на завтра в определенном месте, но расчеты по такой модели продлятся трое суток даже на самом мощном компьютере. Зачем нужна модель, опаздывающая на два дня? Ну а точность модели, используемой синоптиками, мы все знаем сами. Зато расчеты по этой модели занимают всего несколько часов.
Ртак, прежде всего нам надо выбрать уровень абстракции, необходимый для правильного описания реального информационного процесса. Затем следует выделить объекты, участвующие РІ этом процессе, Рё установить СЃРІСЏР·Рё между этими объектами. Как это сделать? Опишите процесс словами Рё проанализируйте получившиеся фразы. "Завод выпускает автомобили". Здесь РґРІР° объекта — завод Рё автомобиль. Производственнотехнические характеристики завода составят набор полей объекта "Завод", Р° процесс выпуска автомобиля будет описан РІ РІРёРґРµ набора методов объекта "Завод".
Пример из другой области: "Преподаватель читает учебный курс". Полями объекта "Преподаватель" будут его фамилия, имя и отчество, научно-педагогический стаж, квалификация, ученая степень, выпущенные им учебники и методические пособия. Методами "Преподавателя" будут такие действия, как "читать", "писать", "повышать квалификацию", "проводить консультацию", "принимать зачет". Полями объекта "Учебный курс" будут его название, программа, количество часов, перечень учебных пособий. Будет ли объект "Учебный курс" обладать какими-то методами или в этом объекте будут только поля? Какие действия выполняет "Учебный курс"? По-видимому, единственным действием объекта "Учебный курс" будет предоставление своих полей другим объектам, значит, нужны методы доступа к полям объекта.
Таким образом, если в словесном описании процесса вам потребовалось сформулировать какое-то понятие, то оно и будет кандидатом на оформление его в виде объекта. Существительные, описывающие это понятие, будут полями объекта, а глаголы — методами будущего объекта.
В объектно-ориентированных языках модель информационного процесса записывается в виде одного или нескольких классов (classes). Каждый класс описывает свойства одного объекта. Класс можно считать проектом, слепком, чертежом, по которому затем будут создаваться конкретные объекты. При описании класса применяются знакомые нам конструкции программирования.
Поля, в которых объект будет хранить необходимую ему информацию, описываются массивами, переменными и константами. Количество переменных и их типы выбираются так, чтобы в наибольшей степени охарактеризовать объект. Они называются полями класса (class fields). Полями класса могут быть не только простые переменные, константы или массивы, но и другие объекты и массивы объектов. Кроме полей в классе можно определить локальные переменные, хранящие промежуточные результаты работы методов класса.
Методы обработки информации, используемые объектом, описываются процедурами и функциями. Они называются методами класса (class methods). Сложные объекты могут содержать несколько десятков методов, а значит, несколько десятков процедур и функций. Методы класса активно используют поля класса, но кроме них могут создавать и свои локальные переменные, необходимые для работы метода.
Кроме полей и методов в классе можно описать и вложенные классы (nested classes), и вложенные интерфейсы, в которые, в свою очередь, можно вложить классы и интерфейсы. Мы можем создать сложную "матрешку" вложенных классов. Поля, методы и вложенные классы первого уровня называются членами класса (class members). Разные школы объектно-ориентированного программирования предлагают разные термины для описания структуры класса, мы используем терминологию, принятую в технологии Java.
Вот набросок описания автомобиля:
class Automobile{
int maxVelocity; // Поле, содержащее наибольшую скорость автомобиля.
int speed; // Поле, содержащее текущую скорость автомобиля.
int weight; // Поле, содержащее вес автомобиля.
// Прочие поля...
void moveTo(int x, int y){
// Метод, моделирующий перемещение автомобиля в точку (x, y).
// Параметры метода x и y — уже не поля, а локальные переменные. int a = 1; // a — локальная переменная, а не поле.
// Тело метода. Здесь описывается способ перемещения автомобиля / / в точку (x, y)
}
// Прочие методы класса...
}
Знатокам Pascal
В Java нет вложенных процедур и функций, в теле метода нельзя описать другой метод.
После того как описание класса закончено, можно создавать конкретные объекты, называемые экземплярами (instances) описанного класса. Создание экземпляров производится в три этапа, подобно описанию массивов (см. главу 1). Сначала объявляются ссылки на объекты: записывается имя класса и после пробела через запятую перечисляются экземпляры класса, точнее, ссылки на них.
Automobile lada2110, fordScorpio, oka;
Затем операцией new определяются сами объекты, под них выделяется оперативная память, ссылка получает адрес этого участка в качестве своего значения.
lada2110 = new Automobile(); fordScorpio = new Automobile(); oka = new Automobile();
РќР° третьем этапе РїСЂРѕРёСЃС…РѕРґРёС‚ инициализация объектов, задаются начальные значения. Ртот этап, как правило, совмещается СЃРѕ вторым, именно для этого РІ операции new повторяется РёРјСЏ класса СЃРѕ скобками Automobile (). Рто так называемый конструктор (constructor) класса, РЅРѕ Рѕ нем РїРѕРіРѕРІРѕСЂРёРј попозже.
Рмена полей, методов Рё вложенных классов Сѓ всех объектов РѕРґРЅРѕРіРѕ класса одинаковы, РѕРЅРё заданы РІ описании класса. Поэтому имена надо уточнять именем ссылки РЅР° объект:
lada2110.maxVelocity = 150; fordScorpio.maxVelocity = 180; oka.maxVelocity = 350; // Почему бы и нет?
oka.moveTo(35, 120);
Напомним, что текстовая строка в кавычках понимается в Java как объект класса String. Поэтому можно написать
int strlen = "Рто объект класса String".length();
Объект "строка" выполняет метод length(), один из методов своего класса String, подсчитывающий количество символов в строке. В результате получаем значение strlen, равное 24. Подобная странная запись встречается в программах, написанных на языке Java, на каждом шагу.
Во многих ситуациях строят несколько моделей с разной степенью детализации. Скажем, для конструирования пальто и шубы нужна менее точная модель контуров человеческого тела и его движений, а для конструирования фрака или вечернего платья — уже гораздо более точная. При этом более точная, с меньшей степенью абстракции, модель будет использовать уже имеющиеся методы менее точной модели.
Рерархия
Не кажется ли вам, что класс Automobile сильно перегружен? Действительно, в мире выпущены миллионы автомобилей разных марок и видов. Что между ними общего, кроме четырех колес? Да и колес может быть больше или меньше. Не лучше ли написать отдельные классы для легковых и грузовых автомобилей, для гоночных автомобилей и вездеходов? Как организовать все это множество классов? На этот вопрос объектноориентированное программирование отвечает так: надо построить иерархию классов.
Рерархия объектов для РёС… классификации используется давно. Особенно детально РѕРЅР° проработана РІ биологии. Р’СЃРµ знакомы СЃ семействами, родами Рё видами. РњС‹ можем сделать описание СЃРІРѕРёС… домашних животных (pets): кошек (cats), собак (dogs), РєРѕСЂРѕРІ (cows) Рё РїСЂ. следующим образом:
class Pet{ // Здесь описываем общие свойства всех домашних любимцев
Master person; // Хозяин животного
int weight, age, eatTime[]; // Вес, возраст, время кормления
int eat(int food, int drink, int time){ // Процесс кормления
// Начальные действия...
if (time == eatTime[i]) person.getFood(food, drink);
// Метод потребления пищи
}
void voice(); // Звуки, издаваемые животным
// Прочее...
}
Затем создаем классы, описывающие более конкретные объекты, связывая их с общим классом Pet:
class Cat extends Pet{ int mouseCatched; void toMouse();
// Прочие свойства
}
class Dog extends Pet{ void preserve();
// Описываются свойства, присущие только кошкам: // число пойманных мышей // процесс ловли мышей
// Свойства собак:
// охранять
Заметьте, что мы не повторяем общие свойства всех домашних животных, описанные в классе Pet. Они наследуются автоматически. Мы можем определить объект класса Dog и использовать в нем все свойства класса Pet так, как будто они описаны в классе Dog. Например, создаем объекты:
Dog tuzik = new Dog(), sharik = new Dog();
После этого определения можно будет написать:
tuzik.age = 3;
int p = sharik.eat(30, 10, 12);
А классификацию можно продолжить так:
class Pointer extends Dog{ ... } // Свойства породы пойнтер
class Setter extends Dog{ ... } // Свойства сеттеров
Заметьте, что на каждом следующем уровне иерархии в класс добавляются новые свойства, но ни одно свойство не пропадает. Поэтому и употребляется слово extends — "расширяет", которое сообщает, что класс Dog — расширение (extension) класса Pet. С другой стороны, количество объектов при этом уменьшается: собак меньше, чем всех домашних животных. Поэтому часто говорят, что класс Dog — подкласс (subclass) класса Pet, а класс Pet — суперкласс (superclass) или надкласс класса Dog.
Часто используют генеалогическую терминологию: родительский класс, дочерний класс, класс-потомок, класс-предок, возникают племянники и внуки, вся беспокойная семейка вступает в отношения, достойные мексиканского сериала.
В этой терминологии говорят о наследовании (inheritance) классов, в нашем примере можно сказать, что класс Dog наследует класс Pet.
Мы еще не определили счастливого владельца нашего домашнего зоопарка. Опишем его в классе Master. Сделаем набросок описания:
class Master{ // Хозяин животного
String name; // Фамилия, имя
// Другие сведения
void getFood(int food, int drink); // Кормление // Прочее...
}
Ответственность
РҐРѕР·СЏРёРЅ Рё его домашние животные постоянно соприкасаются РІ жизни. РС… взаимодействие выражается глаголами "гулять", "кормить", "охранять", "чистить", "ласкаться", "проситься" Рё РїСЂ. Для описания взаимодействия объектов применяется третий принцип объектно-ориентированного программирования — обязанность или ответственность.
В нашем примере рассматривается только взаимодействие в процессе кормления, описываемое методом eat (). В этом методе животное обращается к хозяину, умоляя его применить метод getFood ( ).
Р’ англоязычной литературе РїРѕРґРѕР±РЅРѕРµ обращение описывается словом message. Рто понятие переведено РЅР° СЂСѓСЃСЃРєРёР№ язык напрямую РЅРё Рє чему РЅРµ обязывающим словом "сообщение”. Лучше было Р±С‹ использовать слово "послание", "поручение" или даже "распоряжение". РќРѕ термин "сообщение" устоялся Рё нам придется его применять. Почему же РЅРµ используется словосочетание "вызов метода", ведь РіРѕРІРѕСЂСЏС‚: "Вызов процедуры"? Дело РІ том, что между этими понятиями есть РїРѕ крайней мере три отличия.
□ Сообщение идет к конкретному объекту, знающему метод решения задачи. В примере этот объект — текущее значение переменной person. Объекты одного и того же класса отличаются друг от друга. У каждого объекта свое текущее состояние, свои значения полей класса, и это может повлиять на выполнение метода.
в–Ў РЎРїРѕСЃРѕР± выполнения поручения, содержащегося РІ сообщении, зависит РѕС‚ объекта, которому РѕРЅРѕ послано. Р’ нашем примере этот объект — С…РѕР·СЏРёРЅ животного. РћРґРёРЅ С…РѕР·СЏРёРЅ поставит РјРёСЃРєСѓ СЃ Chappi, РґСЂСѓРіРѕР№ Р±СЂРѕСЃРёС‚ кость, третий выгонит собаку РЅР° улицу. Рто интересное свойство называется полиморфизмом (polymorphism) Рё будет обсуждаться далее.
в–Ў Обращение Рє методу произойдет только РЅР° этапе выполнения программы, компилятор ничего РЅРµ знает РїСЂРѕ метод. Рто называется "РїРѕР·РґРЅРёРј связыванием" РІ противовес "раннему связыванию", РїСЂРё котором процедура присоединяется Рє программе РЅР° этапе РєРѕРјРїРѕРЅРѕРІРєРё.
Ртак, объект sharik, выполняя СЃРІРѕР№ метод eat (), посылает сообщение объекту, ссылка РЅР° который содержится РІ переменной person, СЃ РїСЂРѕСЃСЊР±РѕР№ выдать ему определенное количество еды Рё питья. Сообщение записано РІ строке person.getFood(food, drink).
Ртим сообщением заключается контракт (contract) между объектами, суть которого РІ том, что объект sharik берет РЅР° себя ответственность (responsibility) задать правильные параметры РІ сообщении, Р° РґСЂСѓРіРѕР№ объект — текущее значение экземпляра person — возлагает РЅР° себя ответственность применить метод кормления getFood(), каким Р±С‹ РѕРЅ РЅРё был.
Модульность
Для того чтобы правильно реализовать принцип ответственности, применяется четвертый принцип объектно-ориентированного программирования — модульность (modularity).
Ртот принцип утверждает: каждый класс должен составлять отдельный модуль. Члены класса, Рє которым РЅРµ планируется обращение РёР·РІРЅРµ, должны быть инкапсулированы.
В языке Java инкапсуляция достигается добавлением модификатора private к описанию члена класса. Например:
private int mouseCatched; private String name; private void preserve();
Рти члены классов становятся закрытыми, РёРјРё РјРѕРіСѓС‚ пользоваться только экземпляры того же самого класса, например tuzik может дать поручение sharik.preserve ().
А если в классе Master мы напишем
private void getFood(int food, int drink);
то метод getFood () не будет найден объектами других классов и несчастный sharik не сможет получить пищу.
В противоположность закрытости мы можем объявить некоторые члены класса открытыми, записав вместо слова private модификатор public, например:
public void getFood(int food, int drink);
К таким членам может обратиться любой объект любого класса.
Знатокам C++
В языке Java словами private, public и protected отмечается каждый член класса в отдельности.
Принцип модульности предписывает открывать члены класса только в случае необходимости. Вспомните надпись на железнодорожном переезде: "Нормальное положение шлагбаума — закрытое".
Если же надо обратиться Рє закрытому полю класса, то рекомендуется включить РІ класс специальные методы доступа (access methods), отдельно для чтения этого поля (get method) Рё отдельно для записи РІ это поле (set method). Рмена методов доступа рекомендуется начинать СЃРѕ слов get Рё set, добавляя Рє этим словам РёРјСЏ поля. Для классов Java, используемых как компоненты большого приложения (такие классы-компоненты РІ технологии Java названы JavaBeans), эти рекомендации возведены РІ ранг закона.
В нашем примере класса Master методы доступа к полю name в самом простом виде могут выглядеть так:
public String getName(){ return name;
}
public void setName(String newName){ name = newName;
}
В реальных ситуациях доступ ограничивается разными проверками, особенно в set-методах, меняющих значения полей. Можно проверять тип вводимого значения, задавать диапазон значений, сравнивать со списком допустимых значений. В нашем примере можно ограничить список имен только членами клуба собаководства и сделать в методе проверку на наличие имени в этом списке.
Кроме методов доступа рекомендуется создавать проверочные is-методы, возвращающие логическое значение true или false. Например, в класс Master можно включить метод, проверяющий, задано ли имя хозяина:
public boolean isEmpty(){ return name == null;
}
Рё использовать этот метод для проверки РїСЂРё доступе Рє полю name, например: if (master01.isEmpty()) master01.setName("Рванов");
Ртак, РјС‹ оставляем открытыми только методы, необходимые для взаимодействия объектов. РџСЂРё этом СѓРґРѕР±РЅРѕ спланировать классы так, чтобы зависимость между РЅРёРјРё была наименьшей, как принято говорить РІ теории РћРћРџ, было наименьшее зацепление (low coupling) между классами. РўРѕРіРґР° структура программы сильно упрощается. РљСЂРѕРјРµ того, такие классы СѓРґРѕР±РЅРѕ использовать как строительные блоки для создания РґСЂСѓРіРёС… программ.
Напротив, члены класса должны активно взаимодействовать друг с другом, как говорят, иметь тесную функциональную связность (high cohesion). Для этого в класс следует включать все методы, описывающие поведение моделируемого объекта, и только такие методы, ничего лишнего. Одно из правил достижения сильной функциональной связности, введенное Карлом Либерхером (Karl J. Lieberherr), получило название закона Деметра. Закон гласит: "В методе m () класса а следует использовать только методы класса а, методы классов, к которым принадлежат параметры метода m(), и методы классов, экземпляры которых создаются внутри метода m()".
Объекты, построенные по этим правилам, подобны кораблям, снабженным всем необходимым. Они уходят в автономное плавание, готовые выполнить любое поручение, на которое рассчитана их конструкция.
Будут ли закрытые члены класса доступны его наследникам? Если в классе Pet написано
private Master person;
то можно ли использовать sharik.person? Разумеется, нет. Ведь в противном случае каждый, интересующийся закрытыми полями класса а, может расширить его классом в и просмотреть закрытые поля класса а через экземпляры класса в.
Когда надо разрешить доступ наследникам класса, но нежелательно открывать его всему миру, тогда в Java используется защищенный (protected) доступ, отмечаемый модификатором protected, например объект sharik может обратиться к полю person родительского класса Pet, если в классе Pet это поле описано так:
protected Master person;
Следует сразу сказать, что на доступ к члену класса влияет еще и пакет, в котором находится класс, но об этом мы поговорим в следующей главе.
РР· этого общего схематического описания принципов объектно-ориентированного программирования РІРёРґРЅРѕ, что язык Java позволяет легко воплощать РІСЃРµ эти принципы. Р’С‹ уже поняли, как записать класс, его поля Рё методы, как инкапсулировать члены класса, как сделать расширение класса Рё какими принципами следует РїСЂРё этом пользоваться. Разберем теперь подробнее правила записи классов Рё рассмотрим РёС… дополнительные возможности.
Но, говоря о принципах ООП, я не могу удержаться от того, чтобы не напомнить основной принцип всякого программирования.
Принцип KISS
Основной, базовый и самый великий принцип программирования на любом языке и при любой парадигме — принцип KISS — не нуждается в переводе. Он расшифровывается так:
"Keep It Simple, Stupid!"
Р’ самом деле, чем проще написана программа, тем легче ее отладить, тем лучше РѕРЅР° будет работать. РљСЂРѕРјРµ того, создатели языка проверяют его работу стандартными конструкциями, которые РІС…РѕРґСЏС‚ потом РІ руководства РїРѕ языку. Рти-то проверенные конструкции Рё следует использовать РІ СЃРІРѕРёС… программах. Вычурные Рё запутанные программы показывают глубину мышления Рё остроумие РёС… создателей, РЅРѕ очень неудобны для сопровождения Рё модификации Рё больше подвержены ошибкам.
Вообще говоря, сложность программы не может быть меньше сложности задачи, которую она решает. Надо только постараться, чтобы сложность программы не превысила сложность задачи. Стандартную задачу надо решать стандартными методами, используя стандартные конструкции языка. Нестандартную задачу тоже лучше решить стандартными методами, и только если это не получается, вводить сложные конструкции.
Каждый язык программирования устанавливает свой стиль. Стиль всякого объектноориентированного языка программирования — следование принципам ООП и использование стандартных библиотек классов. Язык Java очень хорошо приспособлен для выражения этого стиля. На нем легко писать программы, следующие стилю ООП.
При изучении языка программирования важно усвоить его стиль и следовать этому стилю в своих программах. Удивительно, но правильный стиль программирования очень помогает в написании правильных эффективных программ.
Упражнения
1. Опишите в виде объекта строительный подъемный кран.
2. Опишите в виде объекта игровой автомат.
3. Смоделируйте в виде объекта сотовый телефон.
Как описать класс и подкласс?
Ртак, описание класса начинается СЃРѕ слова class, после которого записывается РёРјСЏ класса. Соглашения "Code Conventions" рекомендуют начинать РёРјСЏ класса СЃ заглавной Р±СѓРєРІС‹.
Перед словом class можно записать модификаторы класса (class modifiers). Рто РѕРґРЅРѕ РёР· слов public, abstract, final, strictfp. Перед именем вложенного класса можно поставить также модификаторы protected, private, static. Модификаторы класса РјС‹ будем вводить РїРѕ мере изучения языка.
Тело класса, в котором в любом порядке перечисляются поля, методы, конструкторы, вложенные классы и интерфейсы, заключается в фигурные скобки.
При описании поля указывается его тип, затем, через пробел, имя и, может быть, начальное значение после знака равенства, которое допустимо записать константным выражением. Все это уже обсуждалось в главе 1.
Описание поля может начинаться с одного или нескольких необязательных модификаторов public, protected, private, static, final, transient, volatile. Если надо поставить несколько модификаторов, то перечислять их JLS рекомендует в указанном порядке, поскольку некоторые компиляторы требуют определенного порядка записи модификаторов. С модификаторами мы будем знакомиться по мере необходимости.
При описании метода указывается тип возвращаемого им значения или слово void, затем, через пробел, имя метода, потом, в скобках, список параметров. После этого в фигурных скобках расписывается выполняемый метод.
Описание метода может начинаться с модификаторов public, protected, private, abstract,
static, final, synchronized, native, strictfp. Мы будем вводить их по необходимости.
В списке параметров через запятую перечисляются тип и имя каждого параметра. Перед типом какого-либо параметра может стоять модификатор final. Такой параметр нельзя изменять внутри метода. Список параметров может отсутствовать, но скобки сохраняются.
Перед началом работы метода для каждого параметра выделяется ячейка оперативной памяти, в которую копируется значение параметра, заданное при обращении к методу. Такой способ называется передачей параметров по значению. Конкретные значения параметров, переданные методу при обращении к нему, называются аргументами метода. Типы аргументов метода должны быть согласованы с типами соответствующих параметров метода.
В листинге 2.1 показано, как можно оформить метод деления пополам для нахождения корня нелинейного уравнения из листинга 1.5.
Листинг 2.1. Нахождение корня нелинейного уравнения методом бисекции
class Bisection2{
private static double final EPS = 1e-8; // Константа класса.
private double a = 0.0, b = 1.5, root; // Закрытые поля экземпляра.
public double getRoot(){return root;} // Метод доступа к полю root.
private double f(double x){
return x*x*x — 3*x*x + 3; // Можно вернуть и что-нибудь другое.
}
private void bisect(){ // Параметров у метода нет —
// метод работает с полями экземпляра. double y = 0.0; // Локальная переменная — не поле.
do{
root = 0.5 *(a + b); y = f(root);
if (Math.abs(y) < EPS) break;
// Корень найден. Выходим из цикла.
// Если на концах отрезка [a; root] функция имеет разные знаки: if (f(a) * y < 0.0) b = root;
// значит, корень здесь, и мы переносим точку b в точку root.
// В противном случае: else a = root;
// переносим точку a в точку root
// Продолжаем до тех пор, пока [a; b] не станет мал.
} while(Math.abs(b-a) >= EPS);
}
public static void main(String[] args){
Bisection2 b2 = new Bisection2(); b2.bisect();
System.out.println("x = " +
b2.getRoot() + // Обращаемся к корню через метод доступа.
", f() = " +b2.f(b2.getRoot()));
}
}
В описании метода f () сохранен старый процедурный стиль: метод получает аргумент, скопированный в параметр x, обрабатывает его и возвращает результат. Описание метода bisect () выполнено в духе ООП: метод активен, он сам обращается к полям экземпляра b2 и сам заносит результат в нужное поле. Метод bisect () — это внутренний механизм класса Bisection2, поэтому он закрыт (private).
Передача аргументов в метод
При обращении к методу создаются локальные переменные для хранения параметров метода на время его работы. Под них выделяются ячейки оперативной памяти, в которые копируются аргументы метода, заданные при обращении к нему. Локальные переменные-параметры существуют только во время выполнения метода, по окончании его работы переменные уничтожаются и память освобождается.
Теория программирования знает несколько способов передачи аргументов в метод. Чаще всего применяются два способа: передача по значению и передача по ссылке.
Если аргумент передается по значению, то он копируется в локальную переменную-параметр метода, созданную во время выполнения метода и существующую только на время его работы. Сам аргумент при этом остается недоступным для метода и, следовательно, не изменяется им.
При передаче по ссылке в метод поступает не сам аргумент, а его адрес. Метод работает не с копией, а непосредственно с аргументом, обращаясь к нему по адресу, переданному в локальную переменную, и изменяет сам аргумент.
В языке Java, как и в языке С, реализован только один способ — передача аргументов по значению. Например, выполнив следующую программу:
class Dummy1{
private static void f(int a){ a = 5;
}
public static void main(String[] args){
int x = 7;
System.out.println(,,До: " + x); f(x);
System.out.println("После: " + x);
}
}
вы увидите значение 7 и до и после выполнения метода f(), потому что он менял локальную переменную a, а не переменную-аргумент x.
Очень часто у метода встречаются параметры ссылочного типа. В этом случае в локальную переменную-параметр копируется значение аргумента-ссылки, которое, среди прочего, содержит адрес, по которому хранится значение переменной. После этого метод будет работать непосредственно с объектом, на который ссылается аргумент, при помощи локальной копии аргумента-ссылки. Следующий пример поясняет это.
class Dummy2{
private static void f(int[] a){ a[0] = 5;
}
public static void main(String[] args){ int[] x = {7};
System.out.println("До: " + x[0]); f (x);
System.out.println("После: " + x[0]);
}
}
Теперь переменная x — это ссылка на массив, которая копируется в локальную переменную, созданную для параметра a. Ссылка a направляется на тот же массив, что и ссылка x. Она меняет нулевой элемент массива, и мы получаем "До: 7", "После: 5". По-прежнему сделана передача аргумента по значению, но теперь аргумент — это ссылка, и в метод f () передается ссылка, а не объект, на который она направлена.
Передача ссылок по значению приводит иногда к неожиданным результатам. В следующем примере:
class Dummy3{
private static void f(int[] a){ a = new int[]{5};
}
public static void main(String[] args){ int[] x = {7};
System.out.println("До: " + x[0]); f (x);
System.out.println("После: " + x[0]);
}
}
мы опять оба раза увидим на экране число 7. Хотя теперь в методе f() изменилась ссылка на массив — параметр этого метода, а не сам массив, но изменилась копия a ссылки x, а не она сама. Копия a получила новое значение, она направлена на новый массив {5}, но сама ссылка x осталась прежней, она по-прежнему направлена на массив {7}.
Знатокам Pascal и C++
В языке Java применяется только передача аргументов по значению.
Перегрузка методов
РРјСЏ метода, число Рё типы параметров образуют сигнатуру (signature) метода. Компилятор различает методы РЅРµ РїРѕ РёС… именам, Р° РїРѕ сигнатурам. Рто позволяет записывать разные методы СЃ одинаковыми именами, различающиеся числом Рё/или типами параметров.
Замечание
Тип возвращаемого значения не входит в сигнатуру метода, значит, методы не могут различаться только типом результата их работы.
Например, в классе Automobile мы записали метод moveTo (int x, int y), обозначив пункт назначения его географическими координатами. Можно определить еще метод moveTo(String destination) для указания географического названия пункта назначения и обращаться к нему так:
oka.moveTo("РњРѕСЃРєРІР°");
Такое дублирование методов называется РёС… перегрузкой (overloading). Перегрузка методов очень СѓРґРѕР±РЅР° РІ использовании. Вспомните, РІ главе 1 РјС‹ выводили данные любого типа РЅР° экран методом println(), РЅРµ заботясь Рѕ том, данные какого именно типа РјС‹ выводим. РќР° самом деле РјС‹ использовали разные методы СЃ РѕРґРЅРёРј Рё тем же именем println, даже РЅРµ задумываясь РѕР± этом. Конечно, РІСЃРµ эти методы надо тщательно спланировать Рё заранее описать РІ классе. Рто Рё сделано РІ классе Printstream, РіРґРµ представлено около двадцати методов print () Рё println ().
Переопределение методов
Если же записать метод в подклассе с тем же именем, параметрами и типом возвращаемого значения, что и в суперклассе, например:
class Truck extends Automobile{ void moveTo(int x, int y){
// Какие-то действия...
}
// Что-то еще, содержащееся в классе Truck...
}
то он перекроет метод суперкласса.
Определив экземпляр класса Truck, например:
Truck gazel = new Truck();
и записав gazel.moveTo(25, 150), мы обратимся к методу класса Truck. Произойдет переопределение (overriding) метода.
При переопределении метода его сигнатура и тип возвращаемого значения должны полностью сохраняться. Если в подклассе мы изменим тип, количество или порядок следования параметров, то получим новый метод, не переопределяющий метод суперкласса. Если изменим только тип возвращаемого значения, то получим ошибку, которую "заметит" компилятор.
Проверку соответствия сигнатуры переопределяемого метода можно возложить на компилятор, записав перед методом подкласса аннотацию ©Override, как это сделано в листинге 2.2. В этом случае компилятор пошлет на консоль сообщение об ошибке, если сигнатура помеченного метода не будет соответствовать сигнатуре ни одного метода суперкласса с тем же именем.
При переопределении метода права доступа к нему можно только расширить, но не сузить. Открытый метод public должен остаться открытым, защищенный protected может стать открытым, но не может стать закрытым.
Можно ли внутри подкласса обратиться к методу суперкласса? Да, можно, если уточнить имя метода словом super, например super.moveTo(30, 40). Можно уточнить и имя метода, записанного в этом же классе, словом this, например this.moveTo(50, 70), но в данном случае это уже излишне. Таким же образом можно уточнять и совпадающие имена полей, а не только методов.
Данные уточнения РїРѕРґРѕР±РЅС‹ тому, как РјС‹ РіРѕРІРѕСЂРёРј РїСЂРѕ себя "СЏ", Р° РЅРµ "Рван Петрович", Рё РіРѕРІРѕСЂРёРј "отец", Р° РЅРµ "Петр РЎРёРґРѕСЂРѕРІРёС‡".
Реализация полиморфизма в Java
Переопределение методов приводит к интересным результатам. В классе Pet мы описали метод voice (). Переопределим его в подклассах и используем в классе Chorus, как показано в листинге 2.2.
Листинг 2.2. Пример полиморфного метода
abstract class Pet{
abstract void voice();
}
class Dog extends Pet{ int k = 10;
В©Override void voice(){
System.out.println("Gav-gav!");
}
}
class Cat extends Pet{
В©Override void voice(){
System.out.println("Miaou!");
}
}
class Cow extends Pet{
В©Override void voice(){
System.out.println("Mu-u-u!");
}
}
public class Chorus{
public static void main(String[] args){ Pet[] singer = new Pet[3]; singer[0] = new Dog(); singer[1] = new Cat(); singer[2] = new Cow();
for (Pet p: singer) p.voice();
}
}
На рис. 2.1 показан вывод этой программы. Животные поют своими голосами!
Рис. 2.1. Результат выполнения программы Chorus |
Все дело здесь в определении поля singer [ ]. Хотя массив ссылок singer [ ] имеет тип Pet, каждый его элемент ссылается на объект своего типа: Dog, Cat, Cow. При выполнении программы вызывается метод конкретного объекта, а не метод класса, которым определялось имя ссылки. Так в Java реализуется полиморфизм.
Знатокам C++
В языке Java все методы являются виртуальными функциями.
Внимательный читатель заметил в описании класса Pet новое слово abstract. Класс Pet и метод voice () являются абстрактными.
Упражнения
4. Опишите в виде класса строительный подъемный кран.
5. Опишите в виде класса игровой автомат.
6. Смоделируйте в виде класса сотовый телефон.
Абстрактные методы и классы
При описании класса Pet мы не можем задать в методе voice () никакой полезный алгоритм, поскольку у всех животных совершенно разные голоса.
Р’ таких случаях РјС‹ записываем только заголовок метода Рё ставим после закрывающей СЃРїРёСЃРѕРє параметров СЃРєРѕР±РєРё точку СЃ запятой. Ртот метод будет абстрактным (abstract),
что необходимо указать компилятору модификатором abstract.
Если класс содержит хоть один абстрактный метод, то создать его экземпляры, а тем более использовать их, не удастся. Такой класс становится абстрактным, что обязательно надо указать модификатором abstract.
Как же использовать абстрактные классы? Только порождая от них подклассы, в которых переопределены абстрактные методы.
Зачем же нужны абстрактные классы? Не лучше ли сразу написать необходимые классы с полностью определенными методами, а не наследовать их от абстрактного класса? Для ответа снова обратимся к листингу 2.2.
Хотя элементы массива singer[] ссылаются на подклассы Dog, Cat, Cow, но все-таки это переменные типа Pet и ссылаться они могут только на поля и методы, описанные в суперклассе Pet. Дополнительные поля подкласса для них недоступны. Попробуйте обратиться, например, к полю k класса Dog, написав singer[0] .k. Компилятор "скажет", что он не может найти такое поле. Поэтому метод, который реализуется в нескольких подклассах, приходится выносить в суперкласс, а если там его нельзя реализовать, то объявить абстрактным. Таким образом, абстрактные классы группируются на вершине иерархии классов.
Кстати, можно задать пустую реализацию метода, просто поставив пару фигурных скобок, ничего не написав между ними, например:
void voice(){}
Получится полноценный метод, хотя он ничего не делает. Но это искусственное решение, запутывающее структуру класса.
Замкнуть же иерархию можно окончательными классами.
Окончательные члены и классы
Пометив метод модификатором final, можно запретить его переопределение РІ подклассах. Рто СѓРґРѕР±РЅРѕ РІ целях безопасности. Р’С‹ можете быть уверены, что метод выполняет те действия, которые РІС‹ задали. Рменно так определены математические функции sin ( ), cos ( ) Рё РїСЂ. РІ классе Math. РњС‹ уверены, что метод Math.cos(x) вычисляет именно РєРѕСЃРёРЅСѓСЃ числа С…. Разумеется, такой метод РЅРµ может быть абстрактным.
Для полной безопасности поля, обрабатываемые окончательными методами, следует сделать закрытыми (private).
Если пометить модификатором final параметр метода, то его нельзя будет изменить внутри метода.
Если же пометить модификатором final весь класс, то его вообще нельзя будет расширить. Так определен, например, класс Math:
public final class Math{ . . . }
Для переменных модификатор final имеет совершенно РґСЂСѓРіРѕР№ смысл. Если пометить модификатором final описание переменной, то ее значение (Р° РѕРЅРѕ должно быть обязательно задано или здесь же, или РІ блоке инициализации, или РІ конструкторе) нельзя изменить РЅРё РІ подклассах, РЅРё РІ самом классе. Переменная превращается РІ константу. Рменно так РІ языке Java определяются константы:
public final int MIN_VALUE = -1, MAX_VALUE = 9999;
По соглашению "Code Conventions" константы записываются прописными буквами, слова в них разделяются знаком подчеркивания.
Класс Object
На самой вершине иерархии классов Java стоит класс Object.
Если при описании класса мы не указываем никакое расширение, т. е. не пишем слово extends и имя класса за ним, как при описании класса Pet, то Java считает этот класс расширением класса Object, и компилятор дописывает это за нас:
class Pet extends Object{ . . . }
Можно записать это расширение и явно.
Сам же класс Obj ect не является ничьим наследником, от него начинается иерархия любых классов Java. В частности, все массивы — прямые наследники класса Object.
Поскольку такой класс может содержать только общие свойства всех классов, в него включено лишь несколько самых общих методов, например метод equals(), сравнивающий данный объект на равенство с объектом, заданным в аргументе, и возвращающий логическое значение. Его можно использовать так:
Object objl = new Dog(), obj2 = new Cat(); if (obj1.equals(obj2)) ...
Оцените объектно-ориентированный дух этой записи: объект obj 1 активен, он сам сравнивает себя с другим объектом. Можно, конечно, записать и obj2.equals(obji), сделав активным объект obj 2, с тем же результатом.
Как указывалось в главе 1, ссылки можно сравнивать на равенство и неравенство:
obj 1 == obj 2; obj 1 ! = obj 2;
В этом случае сопоставляются адреса объектов, мы можем узнать, не указывают ли обе ссылки на один и тот же объект.
Метод equals () же сравнивает содержимое объектов в их текущем состоянии, фактически он реализован в классе Object как тождество: объект равен только самому себе. Поэтому его обычно переопределяют в подклассах; более того, правильно спроектированные, "хорошо воспитанные" классы должны переопределить методы класса Obj ect, если их не устраивает стандартная реализация. Например, в классе String метод equals () сравнивает не адреса размещения строк в оперативной памяти, а символы, из которых состоит строка, как мы увидим в главе 5.
Второй метод класса Object, часто требующий переопределения,- метод hashCode( ) —
возвращает целое число, уникальное для каждого объекта данного класса, его идентификатор. Рто число позволяет однозначно определить объект. РћРЅРѕ используется РјРЅРѕРіРёРјРё стандартными классами Java. Реализация метода hashCode (), сделанная РІ классе Obj ect, может оказаться недостаточной для какого-то подкласса. Р’ таком случае метод hashCode () следует переопределить.
Третий метод класса Object, который следует переопределять РІ подклассах, — метод tostring (). Рто метод без параметров, который выражает содержимое объекта строкой символов Рё возвращает объект класса string. Р’ классе Object метод tostring() реализован очень СЃРєСѓРґРЅРѕ — РѕРЅ выдает РёРјСЏ класса Рё идентификатор объекта, возвращаемый методом hashCode (). Метод tostring() важен потому, что исполняющая система Java обращается Рє нему каждый раз, РєРѕРіРґР° требуется представить объект РІ РІРёРґРµ строки, например РІ методе println(). Обычно метод tostring() переопределяют так, чтобы РѕРЅ возвращал информацию Рѕ классе объекта Рё текущие значения его полей, записанные РІ РІРёРґРµ строк символов.
Конструкторы класса
Р’С‹ уже обратили внимание РЅР° то, что РІ операции new, определяющей экземпляры класса, повторяется РёРјСЏ класса СЃРѕ скобками. Рто похоже РЅР° обращение Рє методу, РЅРѕ что Р·Р° "метод", РёРјСЏ которого полностью совпадает СЃ именем класса?
Такой "метод" называется конструктором класса (class constructor). Его задача — определение полей создаваемого объекта начальными значениями. Своеобразие конструктора заключается не только в имени. Перечислим особенности конструктора.
□ Конструктор имеется в любом классе. Даже если вы его не написали, компилятор Java сам создаст конструктор по умолчанию (default constructor), который, впрочем, пуст, он не делает ничего, кроме вызова аналогичного конструктора по умолчанию суперкласса.
□ Конструктор выполняется автоматически при создании экземпляра класса, после распределения памяти и инициализации полей, но до начала использования создаваемого объекта.
□ Конструктор не возвращает никакого значения. Поэтому в его описании не пишется даже слово void, но можно задать один из трех модификаторов: public, protected или
private.
□ Конструктор не является методом, он даже не считается членом класса. Поэтому его нельзя наследовать или переопределить в подклассе.
□ Тело конструктора может начинаться:
• с вызова одного из конструкторов суперкласса, для этого записывается слово super () с параметрами конструктора суперкласса в скобках, если они нужны;
• с вызова другого конструктора того же класса, для этого записывается слово this () с параметрами в скобках, если они нужны.
Если же обращение к конструктору суперкласса super () в начале конструктора не написано, то сначала выполняется конструктор суперкласса без аргументов, затем происходит инициализация полей значениями, указанными при их объявлении, а уж потом то, что записано в конструкторе.
Во всем остальном конструктор можно считать обычным методом, в нем разрешается записывать любые операторы, даже оператор return, но только пустой, без всякого возвращаемого значения.
В классе может быть несколько конструкторов. Поскольку у них одно и то же имя, совпадающее с именем класса, то они должны отличаться типом и/или количеством параметров.
Если РІС‹ РЅРµ написали РІ своем классе РЅРё РѕРґРЅРѕРіРѕ конструктора, то компилятор, как уже сказано ранее, создаст конструктор РїРѕ умолчанию. РќРѕ если РІС‹ написали хоть РѕРґРёРЅ конструктор, то компилятор ничего делать РЅРµ будет Рё вам самим надо написать конструктор РїРѕ умолчанию, без аргументов. Рто следует сделать обязательно, если РІС‹ будете расширять СЃРІРѕР№ класс, С‚. Рє. конструкторы подкласса вызывают конструктор РїРѕ умолчанию суперкласса.
В наших примерах мы пока ни разу не рассматривали конструкторы классов, поэтому при создании экземпляров наших классов вызывался конструктор класса Object.
Операция new
Пора подробнее описать операцию с одним операндом, обозначаемую словом new. Она применяется для выделения памяти массивам и объектам.
В первом случае в качестве операнда указывается тип элементов массива и количество его элементов в квадратных скобках, например:
double a[] = new double[100];
Рлементы массива обнуляются.
Во втором случае операндом служит конструктор класса. Если конструктора в классе нет, то вызывается конструктор по умолчанию.
Числовые поля класса получают нулевые значения, логические поля — значение false, ссылки — значение null.
Результатом операции new будет ссылка РЅР° созданный объект. Рта ссылка может быть присвоена переменной типа "ссылка" РЅР° данный тип:
Dog k9 = new Dog();
но может использоваться и непосредственно:
new Dog().voice();
Здесь после создания безымянного объекта сразу выполняется его метод voice (). Такая странная запись встречается в программах, написанных на Java, на каждом шагу. Она возможна потому, что приоритет операции new выше, чем приоритет операции обращения к методу, обозначаемой точкой.
Упражнение
7. Введите конструкторы в классы, определенные в упражнениях 4—6.
Статические члены класса
Разные экземпляры РѕРґРЅРѕРіРѕ класса имеют совершенно независимые РґСЂСѓРі РѕС‚ РґСЂСѓРіР° поля, принимающие различные значения. Рзменение поля РІ РѕРґРЅРѕРј экземпляре никак РЅРµ влияет РЅР° то же поле РІ РґСЂСѓРіРѕРј экземпляре. Р’ каждом экземпляре для таких полей выделяется СЃРІРѕСЏ ячейка памяти. Поэтому такие поля называются переменными экземпляра класса (instance variables) или переменными объекта.
РРЅРѕРіРґР° надо определить поле, общее для всего класса, изменение которого РІ РѕРґРЅРѕРј экземпляре повлечет изменение того же поля РІРѕ всех экземплярах. Например, РјС‹ хотим РІ классе Automobile отмечать порядковый заводской номер автомобиля. Такие поля называются переменными класса (class variables). Для переменных класса выделяется только РѕРґРЅР° ячейка памяти, общая для всех экземпляров. Переменные класса образуются РІ Java модификатором static. Р’ листинге 2.3 РјС‹ записываем этот модификатор РїСЂРё определении переменной number.
Листинг 2.3. Статическое поле
class Automobile!
private static int number;
Automobile(){ number++;
system.out.println("From Automobile constructor:" + " number = " + number);
}
}
public class AutomobileTest{
public static void main(string[] args){
Automobile lada2105 = new Automobile(), fordscorpio = new Automobile(), oka = new Automobile();
}
}
Получаем результат, показанный на рис. 2.2.
Р РёСЃ. 2.2. Рзменение статического поля |
Рнтересно, что Рє статическим полям можно обращаться СЃ именем класса, Automobile. number, Р° РЅРµ только СЃ именем экземпляра, lada2105. number, причем это можно делать, даже если РЅРµ создан РЅРё РѕРґРёРЅ экземпляр класса. Дело РІ том, что поля класса определяются РїСЂРё загрузке файла СЃ классом РІ оперативную память, еще РґРѕ создания экземпляров класса.
Для работы с такими статическими полями обычно создаются статические методы, помеченные модификатором static. Для методов слово static имеет совсем другой смысл, чем для полей, потому что исполняющая система Java всегда создает в памяти только одну копию машинного кода метода, разделяемую всеми экземплярами, независимо от того, статический это метод или нет.
Основная особенность статических методов — РѕРЅРё работают напрямую только СЃРѕ статическими полями Рё методами класса. РџСЂРё попытке обращения статического метода Рє нестатическим полям Рё методам класса РЅР° консоль посылается сообщение РѕР± ошибке. Рто позволяет повысить безопасность программы, предотвратить случайное изменение полей Рё методов отдельных экземпляров класса.
Такая особенность статических методов РїСЂРёРІРѕРґРёС‚ Рє интересному побочному эффекту. РћРЅРё РјРѕРіСѓС‚ выполняться, даже если РЅРµ создан РЅРё РѕРґРёРЅ экземпляр класса. Достаточно уточнить РёРјСЏ метода именем класса (Р° РЅРµ именем объекта), чтобы метод РјРѕРі работать. Рменно так РјС‹ пользовались методами класса Math, РЅРµ создавая его экземпляры, Р° просто записывая
Math.abs (x), Math. sqrt (x). Точно так же мы использовали метод system.out .println ( ) . Да и методом main () мы пользуемся, вообще не создавая никаких объектов.
Поэтому статические методы называются методами класса (class methods), в отличие от нестатических методов, называемых методами экземпляра (instance methods).
Отсюда вытекают другие особенности статических методов:
□ в статическом методе нельзя использовать ссылки this и super;
□ статические методы не могут быть абстрактными;
□ статические методы переопределяются в подклассах только как статические;
□ при переопределении статических методов полиморфизм не действует, ссылки всегда направляются на методы класса, а не объекта.
Рменно РїРѕ этим причинам РІ листинге 1.5 РјС‹ пометили метод f() модификатором static. РќРѕ РІ листинге 2.1 РјС‹ работали СЃ экземпляром b2 класса Bisection2, Рё нам РЅРµ потребовалось объявлять метод f () статическим.
Статические переменные инициализируются еще до начала работы конструктора, но при инициализации можно использовать только константные выражения. Если же инициализация требует сложных вычислений, например циклов для задания значений элементам статических массивов или обращений к методам, то эти вычисления заключают в блок, помеченный словом static, который тоже будет выполнен при загрузке класса, до конструктора:
static int[] a = new a[10]; static{
for (int k: a) a[k] = k * k;
}
Операторы, заключенные в такой блок, выполняются только один раз, при загрузке класса, а не при создании каждого экземпляра.
Здесь внимательный читатель, наверное, поймал меня: "А говорил, что все действия выполняются только с помощью методов!" Каюсь: блоки статической инициализации и блоки инициализации экземпляра записываются вне всяких методов и выполняются до начала выполнения не то что метода, но даже конструктора.
Класс Complex
Комплексные числа широко используются не только в математике. Они часто применяются в графических преобразованиях, в построении фракталов, не говоря уже о фи-
зике и технических дисциплинах. Но класс, описывающий комплексные числа, почему-то не включен в стандартную библиотеку Java. Восполним этот пробел.
Листинг 2.4 длинный, но просмотрите его внимательно, при обучении языку программирования очень полезно чтение программ на этом языке. Более того, только программы и стоит читать, пояснения автора лишь мешают вникнуть в смысл действий (шутка).
Листинг 2.4. Класс Complex
class Complex{
private static final double EPs = 1e-12; // Точность вычислений. private double re, im; // Действительная и мнимая части.
// Четыре конструктора:
Complex(double re, double im){ this.re = re; this.im = im;
}
Complex(double re){this(re, 0.0);}
Complex(){this(0.0, 0.0);}
Complex(Complex z){this(z.re, z.im);}
// Методы доступа: public double getRe(){return re;} public double getIm(){return im;}
public Complex getZ(){return new Complex(re, im);} public void setRe(double re){this.re = re;} public void setIm(double im){this.im = im;} public void setZ(Complex z){re = z.re; im = z.im;}
// Модуль и аргумент комплексного числа: public double mod(){return Math.sqrt(re * re + im * im);} public double arg(){return Math.atan2(re, im);}
// Проверка: действительное число? public boolean isReal(){return Math.abs(im) < EPs;} public void pr(){ // Вывод на экран
system.out.println(re + (im < 0.0 ? "" : "+") + im + "i");
}
// Переопределение методов класса Object: public boolean equals(Complex z){ return Math.abs(re — z.re) < EPs &&
Math.abs(im — z.im) < EPs;
}
public string tostring(){
return "Complex: " + re + " " + im;
}
// Методы, реализующие операции +=, -=, *=, /= public void add(Complex z){re += z.re; im += z.im;} public void sub(Complex z){re -= z.re; im -= z.im;} public void mul(Complex z){
double t = re * z.re — im * z.im; im = re * z.im + im * z.re; re = t;
public void div(Complex z){
double m = z.re * z.re + z.im * z.im; double t = re * z.re — im * z.im; im = (im * z.re — re * z.im) / m; re = t / m;
}
// Методы, реализующие операции +, -, *, /
public Complex plus(Complex z){
return new Complex(re + z.re, im + z.im);
}
public Complex minus(Complex z){
return new Complex(re — z.re, im — z.im);
}
public Complex asterisk(Complex z){ return new Complex(
re * z.re — im * z.im, re * z.im + im * z.re);
}
public Complex slash(Complex z){
double m = z.re * z.re + z.im * z.im; return new Complex(
(re * z.re — im * z.im) / m, (im * z.re — re * z.im) / m);
}
}
// Проверим работу класса Complex. public class ComplexTest{
public static void main(string[] args){ Complex z1 = new Complex(),
z2 = new Complex(1.5),
z3 = new Complex(3.6, -2.2),
z4 = new Complex(z3);
// Оставляем пустую строку. "); z1.pr();
"); z2.pr();
"); z3.pr();
"); z4.pr();
// Работает метод toString().
System.out.println(); system.out.print("z1 system.out.print("z2 system.out.print("z3 system.out.print("z4 System.out.println(z4); z2.add(z3);
'); z2.pr(); '); z2.pr(); '); z2.pr(); '); z3.pr();
system.out.print("z2 + z3 z2.div(z3);
system.out.print("z2 / z3 z2 = z2.plus(z2); system.out.print("z2 + z2 z3 = z2.slash(z1); system.out.print("z2 / z1
}
На рис. 2.3 показан вывод этой программы.
Рис. 2.3. Вывод программы ComplexTest |
Метод main()
Всякая программа, оформленная как приложение (application), должна содержать метод с именем main. Он может быть один на все приложение или присутствовать в некоторых классах этого приложения, а может находиться и в каждом классе.
Метод main () записывается как обычный метод, может содержать любые описания и действия, но он обязательно должен быть открытым (public), статическим (static), не иметь возвращаемого значения (void). У него один параметр, которым обязательно должен быть массив строк (string [ ]). По традиции этот массив называют args, хотя имя может быть любым.
Рти особенности возникают РёР·-Р·Р° того, что метод main() вызывается автоматически исполняющей системой Java РІ самом начале выполнения приложения, РєРѕРіРґР° еще РЅРµ создан РЅРё РѕРґРёРЅ объект. РџСЂРё вызове интерпретатора java указывается класс, РіРґРµ записан метод main (), СЃ которого надо начать выполнение. Поскольку классов СЃ методом main() может быть несколько, допустимо построить приложение СЃ дополнительными точками РІС…РѕРґР°, начиная выполнение приложения РІ разных ситуациях РёР· различных классов.
Часто метод main() заносят в каждый класс с целью отладки. В этом случае в метод main () включают тесты для проверки работы всех методов класса.
РџСЂРё вызове интерпретатора java можно передать РІ метод main() несколько аргументов, которые интерпретатор заносит РІ массив строк. Рти аргументы перечисляются РІ строке вызова java через пробел сразу после имени класса. Если же аргумент содержит пробелы, надо заключить его РІ кавычки. Кавычки РЅРµ Р±СѓРґСѓС‚ включены РІ аргумент, это только ограничители.
Все это легко понять на примере листинга 2.5, в котором записана программа, просто выводящая на консоль аргументы, передаваемые в метод main () при запуске.
Листинг 2.5. Передача аргументов в метод main()
class Echo{
public static void main(string[] args){ for (string s: args)
system.out.println("arg = " + s);
}
}
На рис. 2.4 показаны результаты работы этой программы с разными вариантами задания аргументов.
Рис. 2.4. Вывод параметров командной строки |
Как видите, имя класса не входит в число аргументов. Оно и так известно в методе
main().
Знатокам C/C++
Поскольку в Java имя файла всегда совпадает с именем класса, содержащего метод main (), оно не заносится в args [0]. Вместо параметра argc используется переменная args. length, имеющаяся в каждом массиве. Доступ к переменным среды разрешен не всегда и осуществляется другим способом. Некоторые значения переменных среды можно просмотреть так:
system.getProperties().list(system.out);
Методы с переменным числом аргументов
Как РІРёРґРЅРѕ РёР· СЂРёСЃ. 2.4, РїСЂРё вызове программы РёР· командной строки РјС‹ можем задавать ей разное число аргументов. Рсполняющая система Java создает массив этих аргументов Рё передает его методу main(). Такую же конструкцию можно сделать РІ своей программе:
class VarArgs{
private static int[] argsl = {1, 2, 3, 4, 5, 6};
private static int[] args2 = {100, 90, 80, 70};
public static int sum(int[] args){ int result = 0;
for (int k: args) result += k; return result;
}
public static void main(string[] args){
System.out.println("Sum1 = " + sum(args1));
System.out.println("Sum2 = " + sum(args2));
}
}
Начиная с пятой версии Java эту конструкцию упростили. Теперь нет необходимости заранее формировать массив аргументов, это сделает компилятор. Программисту достаточно записать в списке параметров метода три точки подряд после имени параметра вместо квадратных скобок, обозначающих массив.
public static int sum(int... args){ int result = 0;
for (int k: args) result += k; return result;
}
При обращении к методу мы просто записываем нужное число аргументов через запятую.
public static void main(string[] args){
System.out.println("Sum1 = " + sum(1, 2, 3, 4, 5, 6));
System.out.println("Sum2 = " + sum(100, 90, 80, 70));
}
Где видны переменные
Р’ языке Java нестатические переменные разрешено объявлять РІ любом месте РєРѕРґР° между операторами. Статические переменные РјРѕРіСѓС‚ быть только полями класса, Р° значит, РЅРµ должны объявляться внутри методов Рё блоков. Какова же область видимости (scope) переменных? РР· каких методов РјС‹ можем обратиться Рє той или РёРЅРѕР№ переменной? Р’ каких операторах использовать? Рассмотрим РЅР° примере листинга 2.6 разные случаи объявления переменных.
Листинг 2.6. Видимость и инициализация переменных
class ManyVariables{
static int x = 9, y; // Статические переменные — поля класса.
// Они известны во всех методах и блоках класса. // Поле y получает значение 0.
static{ // Блок инициализации статических переменных.
// Выполняется РѕРґРёРЅ раз РїСЂРё первой загрузке класса // после инициализаций РІ объявлениях переменных. x = 99; // Ртот оператор выполняется РІ блоке РІРЅРµ РІСЃСЏРєРѕРіРѕ метода!
int a = 1, p; // Нестатические переменные — поля экземпляра.
// Рзвестны РІРѕ всех методах Рё блоках класса,
// в которых они не перекрыты другими переменными // с тем же именем.
// Поле p получает значение 0.
{ // Блок инициализации экземпляра.
// Выполняется РїСЂРё создании каждого экземпляра // после инициализаций РїСЂРё объявлениях переменных. p = 999; // Ртот оператор выполняется РІ блоке РІРЅРµ РІСЃСЏРєРѕРіРѕ метода!
}
static void f(int b){ // Параметр метода b — локальная переменная,
// известная только внутри метода. int a = 2; // Рто вторая переменная СЃ тем же именем "a".
// Она известна только внутри метода f()
// и здесь перекрывает первую "a".
int c; // Локальная переменная, известна только в методе f().
// Не получает никакого начального значения // и должна быть определена перед применением.
{ int c = 555; // Сшибка! Попытка повторного объявления.
int x = 333; // Локальная переменная, известна только в этом блоке.
}
// Здесь переменная x уже неизвестна. for (int d = 0; d < 10; d++){
// Переменная цикла d известна только в цикле. int a = 4; // Ошибка!
int e = 5; // Локальная переменная, известна только в цикле for.
e++; // Рнициализируется РїСЂРё каждом выполнении цикла.
System.out.println("e = " + e); // Выводится всегда "e = 6".
}
// Здесь переменные d и e неизвестны.
}
public static void main(string[] args){
int a = 9999; // Локальная переменная, известна только внутри
// метода main().
f (a) ;
}
}
Обратите внимание на то, что переменные класса и экземпляра неявно присваивают нулевые значения. Символы неявно получают значение ’ \u0000 ’, логические переменные — значение false, ссылки получают неявно значение null.
Локальные же переменные неявно не инициализируются. Они должны либо явно присваивать значения, либо определяться до первого использования. К счастью, компилятор замечает неопределенные локальные переменные и сообщает о них.
Внимание!
Поля класса при объявлении обнуляются, локальные переменные автоматически не инициализируются.
Р’ листинге 2.6 появилась еще РѕРґРЅР° новая конструкция: блок инициализации экземпляра (instance initialization). Рто просто блок операторов РІ фигурных скобках, РЅРѕ записывается РѕРЅ РІРЅРµ РІСЃСЏРєРѕРіРѕ метода, РїСЂСЏРјРѕ РІ теле класса. Ртот блок выполняется РїСЂРё создании каждого экземпляра, после static-блоков Рё инициализации РїСЂРё объявлении переменных, РЅРѕ РґРѕ выполнения конструктора. РћРЅ играет такую же роль, как Рё static-блок для статических переменных. Зачем же РѕРЅ нужен, ведь РІСЃРµ его содержимое можно написать РІ начале конструктора? РћРЅ применяется РІ тех случаях, РєРѕРіРґР° конструктор написать нельзя, Р° именно РІ безымянных внутренних классах.
Вложенные классы
Р’ этой главе уже несколько раз упоминалось, что РІ теле класса можно сделать описание РґСЂСѓРіРѕРіРѕ, вложенного (nested) класса. Рђ РІРѕ вложенном классе можно СЃРЅРѕРІР° описать вложенный, внутренний (inner) класс Рё С‚. Рґ. Рта "матрешка" кажется вполне естественной, РЅРѕ РІС‹ уже поднаторели РІ написании классов, Рё Сѓ вас возникает масса РІРѕРїСЂРѕСЃРѕРІ.
□ Можем ли мы из вложенного класса обратиться к членам внешнего класса? Можем, для того это все и задумывалось.
□ А можем ли мы в таком случае определить экземпляр вложенного класса, не определяя экземпляры внешнего класса? Нет, не можем, сначала надо определить хоть один экземпляр внешнего класса, матрешка ведь!
в–Ў Рђ если экземпляров внешнего класса несколько, как узнать, СЃ каким экземпляром внешнего класса работает данный экземпляр вложенного класса? РРјСЏ экземпляра вложенного класса уточняется именем связанного СЃ РЅРёРј экземпляра внешнего класса. Более того, РїСЂРё создании вложенного экземпляра операция new тоже уточняется именем внешнего экземпляра.
в–Ў Рђ?..
Хватит вопросов, давайте разберем все по порядку.
Все вложенные классы можно разделить на вложенные классы-члены класса (member classes), описанные вне методов, и вложенные локальные классы (local classes), описанные внутри методов и/или блоков. Локальные классы, как и все локальные переменные, не являются членами класса.
Классы-члены РјРѕРіСѓС‚ быть объявлены статическими СЃ помощью модификатора static. Поведение статических классов-членов ничем РЅРµ отличается РѕС‚ поведения обычных классов, отличается только обращение Рє таким классам. Поэтому РѕРЅРё называются вложенными классами верхнего СѓСЂРѕРІРЅСЏ (nested top-level classes), хотя статические классы-члены можно вкладывать РґСЂСѓРі РІ РґСЂСѓРіР°. Р’ РЅРёС… можно объявлять статические члены. Рспользуются РѕРЅРё обычно для того, чтобы сгруппировать вспомогательные классы вместе СЃ основным классом.
Все нестатические вложенные классы называются внутренними (inner). В них нельзя объявлять статические члены.
Локальные классы, как и все локальные переменные, известны только в блоке, в котором они определены. Они могут быть безымянными (anonymous classes).
В листинге 2.7 рассмотрены все эти случаи.
Листинг 2.7. Вложенные классы
class Nested{
static private int pr; // Переменная pr объявлена статической,
// чтобы к ней был доступ из статических классов A и AB.
String s = "Member of Nested";
// Вкладываем статический класс.
static class A{ // Полное имя этого класса — Nested.A
private int a = pr;
String s = "Member of A";
// Во вложенный класс A вкладываем еще один статический класс. static class AB{ // Полное имя класса — Nested.A.AB
private int ab = pr;
String s = "Member of AB";
}
}
// В класс Nested вкладываем нестатический класс. class B{ // Полное имя этого класса — Nested.B
private int b = pr;
String s = "Member of B";
// В класс B вкладываем еще один класс.
class BC{ // Полное имя класса — Nested.B.BC
private int bc = pr;
String s = "Member of BC";
}
void f(final int i){ // Без слова final переменные i и j
// нельзя использовать в локальном классе D.
final int j = 99;
class D{ // Локальный класс D известен только внутри f().
private int d = pr;
String s = "Meimoer of D"; void pr(){
// Обратите внимание на то, как различаются // переменные с одним и тем же именем "s". System.out.println(s + (i+j)); // "s" эквивалентно "this.s".
System.out.println(B.this.s);
System.out.println(Nested.this.s);
// System.out.println(AB.this.s); // Нет доступа.
// System.out.println(A.this.s); // Нет доступа.
}
}
D d = new D(); // Объект определяется тут же, в методе f().
d.pr(); // Объект известен только в методе f().
}
}
void m(){
new Object(){ // Создается объект безымянного класса,
// указывается конструктор его суперкласса.
private int e = pr; void g(){
System.out.println("From g()");
}
}.g(); // Тут же выполняется метод только что созданного объекта.
}
}
public class NestedClasses{
public static void main(String[] args){
Nested nest = new Nested(); // Последовательно раскрываются
// три матрешки.
Nested.A theA = nest.new A(); // Полное имя класса и уточненная
// операция new. Но конструктор только вложенного класса.
Nested.A.AB theAB = theA.new AB(); // Те же правила.
// Операция new уточняется только одним именем.
Nested.B theB = nest.new B(); // Еще одна матрешка.
Nested.B.BC theBC = theB.new BC();
theB.f(999); // Методы вызываются обычным образом.
nest.m();
}
}
Ну как? Поняли что-нибудь? Если вы все поняли и готовы применять эти конструкции в своих программах, значит, вы можете перейти к следующему разделу. Если ничего не поняли, значит, вы — нормальный человек. Помните принцип KISS и используйте вложенные классы как можно реже.
Теперь дадим пояснения.
в–Ў Как видите, доступ Рє полям внешнего класса Nested возможен отовсюду, даже Рє закрытому полю pr. Рменно для этого РІ Java Рё введены вложенные классы. Остальные конструкции добавлены вынужденно, для того чтобы увязать концы СЃ концами.
□ Язык Java позволяет использовать одни и те же имена в разных областях видимости -поэтому пришлось уточнять константу this именем класса: Nested.this, B.this.
□ В безымянном классе не может быть конструктора, ведь имя конструктора должно совпадать с именем класса, — поэтому пришлось использовать имя суперкласса, в примере это класс Object. Вместо конструктора в безымянном классе используется блок инициализации экземпляра, о котором говорилось в предыдущем разделе.
□ Нельзя создать экземпляр вложенного класса, не создав предварительно экземпляр внешнего класса, — поэтому пришлось подстраховать это правило уточнением операции new именем экземпляра внешнего класса nest. new, theA. new, theB. new.
□ При определении экземпляра указывается полное имя вложенного класса, но в операции new записывается просто конструктор класса.
Введение вложенных классов сильно усложнило синтаксис Рё поставило РјРЅРѕРіРѕ задач разработчикам языка. Рто еще РЅРµ РІСЃРµ. Дотошный читатель уже зарядил РЅРѕРІСѓСЋ РѕР±РѕР№РјСѓ РІРѕРїСЂРѕСЃРѕРІ.
□ Можно ли наследовать вложенные классы? Можно.
□ Как из подкласса обратиться к методу суперкласса? Константа super уточняется именем соответствующего суперкласса, подобно константе this.
□ А могут ли вложенные классы быть расширениями других классов? Могут.
□ А как?.. Помните принцип KISS!!!
Механизм вложенных классов станет понятнее, если посмотреть, какие файлы с байткодами создал компилятор:
□ Nested$1$D.class — локальный класс D, вложенный в класс Nested;
□ Nested$1.class — безымянный класс;
□ Nested$A$AB.class — класс Nested.A.AB;
□ Nested$A.class — класс Nested.A;
□ Nested$B$BC.class — класс Nested.B.BC;
□ Nested$B.class — класс Nested.B;
□ Nested.class — внешний класс Nested;
□ NestedClasses.class — класс с методом main ().
Компилятор разложил "матрешки" и, как всегда, создал отдельные файлы для каждого класса. При этом, поскольку в идентификаторах недопустимы точки, компилятор заменил их знаками доллара. Для безымянного класса компилятор придумал имя. Локальный класс компилятор пометил номером.
Оказывается, вложенные классы существуют только на уровне исходного кода. Виртуальная машина Java ничего не знает о вложенных классах. Она работает с обычными внешними классами. Для взаимодействия объектов вложенных классов компилятор вставляет в них специальные закрытые поля. Поэтому в локальных классах можно использовать только константы объемлющего метода, т. е. переменные, помеченные словом final. Виртуальная машина просто не догадается передавать изменяющиеся значения переменных в локальный класс. Таким образом, не имеет смысла помечать вложенные классы модификатором private, все равно они выходят на самый внешний уровень.
Все эти вопросы можно не брать в голову. Вложенные классы в Java используются, как правило, только в самом простом виде, главным образом при обработке событий, возникающих при действиях с мышью и клавиатурой.
Р’ примере СЃ домашними животными РјС‹ сделали объект person класса Master — владелец животного — полем класса Pet. Если класс Master больше РЅРёРіРґРµ РЅРµ используется, то можно определить его РїСЂСЏРјРѕ внутри класса Pet, сделав класс Master вложенным (inner) классом. Рто выглядит следующим образом:
class Pet{
// В этом классе описываем общие свойства всех домашних любимцев. class Master{
// Хозяин животного. string name; // Фамилия, имя.
// Другие сведения...
void getFood(int food, int drink); // Кормление.
// Прочее...
}
int weight; // Вес животного.
int age; // Возраст животного.
Date eatTime[]; // Массив, содержащий время кормления.
int eat(int food, int drink, Date time){ // Процесс кормления.
// Начальные действия.
if (time == eatTime[i]) person.getFood(food, drink);
// Метод потребления пищи...
}
void voice(); // Звуки, издаваемые животным.
// Прочее.
}
Вложение класса удобно тем, что методы внешнего класса могут напрямую обращаться к полям и методам вложенного в него класса. Но ведь того же самого можно было добиться по-другому. Может, следовало расширить класс Master, сделав класс Pet его наследником?
В каких же случаях следует создавать вложенные классы, а когда лучше создать иерархию классов? В теории ООП вопрос о создании вложенных классов решается при рассмотрении отношений "быть частью" и "являться".
Отношения "быть частью" и "являться"
Теперь у нас появились две различные иерархии классов. Одну иерархию образует наследование классов, другую — вложенность классов.
Определив, какие классы будут написаны в вашей программе и сколько их будет, подумайте, как спроектировать взаимодействие классов. Вырастить пышное генеалогическое дерево классов-наследников или расписать "матрешку" вложенных классов?
Теория ООП советует прежде всего выяснить, в каком отношении находятся объекты классов Master и Pet — в отношении "класс Master является экземпляром класса Pet" или в отношении "класс Master является частью класса Pet". Скажем, "собака является животным" или "собака является частью животного"? Другой пример: "мотор является автомобилем" или "мотор является частью автомобиля"? Ясно, что собака — животное и в этой ситуации надо выбрать наследование, но мотор — часть автомобиля и здесь надо выбрать вложение.
Отношение "класс А является экземпляром класса В" по-английски записывается как "a class A is a class B", поэтому в теории ООП называется отношением "is-a". Отношение же "класс А является частью класса В" по-английски "a class A has a class B", и такое отношение называется отношением "has-a".
Отношение "is-a" — это отношение "обобщение-детализация", отношение большей и меньшей абстракции, и ему соответствует наследование классов.
Отношение "has-a" — это отношение "целое-часть" и ему соответствует вложение классов.
Вернемся к нашим животным и их хозяевам и постараемся ответить на вопрос: "класс Master является экземпляром класса Pet" или "класс Master является частью класса Pet"? Ясно, что не верно ни то, ни другое. Классы Master и Pet не связаны ни тем, ни другим образом. Поэтому мы и сделали объект класса Master полем класса Pet.
Заключение
После прочтения этой главы вы получили представление о современной парадигме программирования — объектно-ориентированном программировании и реализации этой парадигмы в языке Java. Если вас заинтересовало ООП, обратитесь к специальной литературе [3—6].
Не беда, если вы не усвоили сразу принципы ООП. Для выработки "объектного" взгляда на программирование нужны время и практика. Части II и III книги как раз и дадут вам эту практику. Но сначала необходимо ознакомиться с важными понятиями языка Java — пакетами и интерфейсами.
Вопросы для самопроверки
1. Какие парадигмы возникали в программировании по мере его развития?
2. Какова современная парадигма программирования?
3. Что такое объектно-ориентированное программирование?
4. Что понимается под объектом в ООП?
5. Каковы основные принципы ООП?
6. Что такое класс в ООП?
7. Какая разница между объектом и экземпляром класса?
8. Что входит в класс Java?
9. Что такое конструктор класса?
10. Какая операция выделяет оперативную память для объекта?
11. Что такое суперкласс и подкласс?
12. Как реализуется полиморфизм в Java?
13. Для чего нужны статические поля и методы класса?
14. Какую роль играют абстрактные методы и классы?
15. Можно ли записать конструктор в абстрактном классе?
16. Почему метод main() должен быть статическим?
17. Почему метод main() должен быть открытым?
ГЛАВА 3