Язык программирования C++ для профессионалов

       

Абстрактные классы


Многие классы сходны с классом employee тем, что в них можно дать разумное определение виртуальным функциям. Однако, есть и другие классы. Некоторые, например, класс shape, представляют абстрактное понятие (фигура), для которого нельзя создать объекты. Класс shape приобретает смысл только как базовый класс в некотором производном классе. Причиной является то, что невозможно дать осмысленное определение виртуальных функций класса shape:

class shape { // ... public: virtual void rotate(int) { error("shape::rotate"); } virtual void draw() { error("shape::draw"): } // нельзя ни вращать, ни рисовать абстрактную фигуру // ... };

Создание объекта типа shape (абстрактной фигуры) законная, хотя совершенно бессмысленная операция:

shape s; // бессмыслица: ``фигура вообще''

Она бессмысленна потому, что любая операция с объектом s приведет к ошибке.

Лучше виртуальные функции класса shape описать как чисто виртуальные. Сделать виртуальную функцию чисто виртуальной можно, добавив инициализатор = 0:

class shape { // ... public: virtual void rotate(int) = 0; // чисто виртуальная функция virtual void draw() = 0; // чисто виртуальная функция };

Класс, в котором есть виртуальные функции, называется абстрактным. Объекты такого класса создать нельзя:

shape s; // ошибка: переменная абстрактного класса shape

Абстрактный класс можно использовать только в качестве базового для другого класса:

class circle : public shape { int radius; public: void rotate(int) { } // нормально: // переопределение shape::rotate void draw(); // нормально: // переопределение shape::draw

circle(point p, int r); };

Если чисто виртуальная функция не определяется в производном классе, то она и остается таковой, а значит производный класс тоже является абстрактным. При таком подходе можно реализовывать классы поэтапно:

class X { public: virtual void f() = 0; virtual void g() = 0; };

X b; // ошибка: описание объекта абстрактного класса X

class Y : public X { void f(); // переопределение X::f };



Содержание раздела