Навигация
Главная
Поиск
Форум
FAQ's
Ссылки
Карта сайта
Чат программистов

Статьи
-Delphi
-C/C++
-Turbo Pascal
-Assembler
-Java/JS
-PHP
-Perl
-DHTML
-Prolog
-GPSS
-Сайтостроительство
-CMS: PHP Fusion
-Инвестирование

Файлы
-Для программистов
-Компонеты для Delphi
-Исходники на Delphi
-Исходники на C/C++
-Книги по Delphi
-Книги по С/С++
-Книги по JAVA/JS
-Книги по Basic/VB/.NET
-Книги по PHP/MySQL
-Книги по Assembler
-PHP Fusion MOD'ы
-by Kest
Professional Download System
Реклама
Услуги

Автоматическое добавление статей на сайты на Wordpress, Joomla, DLE
Заказать продвижение сайта
Программа для рисования блок-схем
Инженерный калькулятор онлайн
Таблица сложения онлайн
Популярные статьи
OpenGL и Delphi... 65535
Форум на вашем ... 65535
21 ошибка прогр... 65535
HACK F.A.Q 65535
Бип из системно... 65535
Гостевая книга ... 65535
Invision Power ... 65535
Пример работы с... 65535
Содержание сайт... 65535
ТЕХНОЛОГИИ ДОСТ... 65535
Организация зап... 65535
Вызов хранимых ... 65535
Создание отчето... 65535
Имитационное мо... 65535
Программируемая... 65535
Эмулятор микроп... 65535
Подключение Mic... 65535
Создание потоко... 65535
Приложение «Про... 65535
Оператор выбора... 65535
Реклама
Сейчас на сайте
Гостей: 14
На сайте нет зарегистрированных пользователей

Пользователей: 13,372
новичок: vausoz
Новости
Реклама
Выполняем курсовые и лабораторные по разным языкам программирования
Подробнее - курсовые и лабораторные на заказ
Delphi, Turbo Pascal, Assembler, C, C++, C#, Visual Basic, Java, GPSS, Prolog, 3D MAX, Компас 3D
Заказать программу для Windows Mobile, Symbian

Моделирование работы перекрёстка по регулированию движения на GPSS + Поя...
Моделирование автовокзала + Отчет + Блок схема
Принадлежит ли точка пересечению двух окружностей на Turbo Pascal + Отче...

Виртуальные функции в конструкторах и деструкторах
Внутри конструкторов и деструкторов динамическое связывание не работает, хотя вызов виртуальных функций не запрещен. Обычно виртуальный вызов означает вызов метода наследника посредством ссылки или указателя на базовый класс. Однако в конструкторах и деструкторах вызывается всегда «родная» функция.
В конструкторах такое поведение оправдано тем, что код конструктора ничего не знает о производных классах. Во время работы конструктора были созданы объекты базовых классов, однако до производных классов дело еще не дошло. Данный конструктор сам мог быть вызван из конструктора производного класса, чтобы создать и проинициализировать объект своего класса. Однако конструктор не знает, кто его вызвал и зачем. Если бы вызов был виртуальным, то были бы возможны ситуации использования неинициализированных переменных производных классов.
В деструкторах ситуация несколько другая — виртуальный вызов опасен тем, что возможен вызов метода уже уничтоженного объекта. Поэтому даже в виртуальном деструкторе вызывается всегда «родной» метод.

Аналогичное решение принято и в языках Smalltalk и Pithon.
Платой за управляемую виртуальность является расход памяти (дополнительные 4 байта на указатель) и времени (инициализация таблицы виртуальных функций).
Чистые виртуальные функции
Когда мы создаем иерархию классов, вершиной иерархии становится класс, в котором перечислены максимально общие свойства всех потомков. Однако в каждом потомке эти свойства — разные. В качестве примера рассмотрим музыкальные инструменты. Инструменты бывают, например, духовыми, струнными и ударными, а духовые инструменты — деревянными и медными. Например, скрипка — струнный инструмент, а флейта — деревянный духовой. Схема иерархии классов может быть такой, как показано в листинге 9.5.

Листинг 9.5. Иерархия музыкальных инструментов
class Instrument; // общий базовый класс
class Wind: public Instrument; // духовые
class Brass: public Wind; // медные
class trombone: public Brass; // тромбон
class Woodwind: public Wind; // деревянные
class flute: public Woodwind; // фпейта
class Stringer: public Instrument; // струнные
class violin: public Stringer; // скрипка
class Percussion: public Instrument; // ударные
class cymbals : public Percussion; // тарелки



Все инструменты имеют некоторые общие свойства, например:
• каждый инструмент как-то конкретно называется;
• на каждом инструменте как-то играют.

Кроме того, все струнные инструменты настраивают. Однако для каждого конкретного инструмента эти свойства разные: на скрипке играют совсем не так, как на флейте, и настройка гитары сильно отличается от настройки фортепиано. Понятно, что функции, реализующие эти свойства, должны быть виртуальными. Здесь возникает одна небольшая проблема: как реализовать эти функции в базовом классе Instrument? Такие общие функции не могут ничего делать в базовом классе — вся работа должна выполняться в производных классах. Единственная возможность добиться этого — определить виртуальные функции с пустым телом, а в производных классах переопределить их. Например, функция для игры может быть такой:
enum notr { Cmiddle, Csharp, Cflat }; // ноты до, до-диез, до-бемоль
virtual void play(note) const {} // играть ноту
Тогда функция настройки (в базовом классе St г i nger для струнных инструментов) может выглядеть так:
virtual void adiustO const {} // настройка



Решение, конечно, приемлемое, но ... «пустое» тело на самом деле не означает отсутствия выполняемых команд. Обычно в оттранслированной функции при входе выполняется так называемый стандартный пролог, а при выходе — стандартный эпилог. Поэтому даже «пустое» тело при выполнении требует накладных расходов.
Для нашего общего предка хорошо подошли бы функции, на самом деле не имеющие тела. Набор таких функций мог бы обозначать только общий интерфейс класса1 (контракт класса), а конкретная их реализация может быть выполнена позже — в классах-наследниках. Отсутствие тела нужно как-то обозначить, ведь мы не можем написать просто прототип — это требует определения функции.
Осознав проблему, Б. Страуструп поступил очень просто: он обозначил отсутствие тела функции нулем! Называется такая виртуальная функция «чистой» (риге) и определяется следующим образом:
virtual тип имя(параметры) = 9;



Несмотря на то, что виртуальная функция — чистая, для такого класса все равно создается таблица виртуальных функций. Страуструп Б. не стал вводить в С++ новое ключевое слово вроде риге ИЛИ abstract, чтобы обозначить отсутствие тела функции, — вместо этого присвоением нуля он подчеркнул, что в таблице виртуальных функций адрес этой функции равен нулю, поэтому вызвать ее нельзя.
Класс, в котором есть хотя бы одна чистая виртуальная функция, называется абстрактным (см. п. 10.4 в [1]). Абстрактный класс отличается от «нормального» класса тем, что объект абстрактного класса создать нельзя, даже динамически операцией new. И при передаче параметра в'функцию невозможно передать объект абстрактного класса по значению — копию-то создать нельзя! Однако указатели (и ссылки) определять можно, так как для указателя размер класса не важен.
Опубликовал Kest November 21 2013 23:59:27 · 0 Комментариев · 3257 Прочтений · Для печати

• Не нашли ответ на свой вопрос? Тогда задайте вопрос в комментариях или на форуме! •


Комментарии
Нет комментариев.
Добавить комментарий
Имя:



smiley smiley smiley smiley smiley smiley smiley smiley smiley
Запретить смайлики в комментариях

Введите проверочный код:* =
Рейтинги
Рейтинг доступен только для пользователей.

Пожалуйста, залогиньтесь или зарегистрируйтесь для голосования.

Нет данных для оценки.
Гость
Имя

Пароль



Вы не зарегистрированны?
Нажмите здесь для регистрации.

Забыли пароль?
Запросите новый здесь.
Поделиться ссылкой
Фолловь меня в Твиттере! • Смотрите канал о путешествияхКак приготовить мидии в тайланде?
Загрузки
Новые загрузки
iChat v.7.0 Final...
iComm v.6.1 - выв...
Visual Studio 200...
CodeGear RAD Stud...
Шаблон для новост...

Случайные загрузки
MPTools
Программа рисует ...
ComboBox97
CwstatusBar
База Allsubmitter...
AlnComponents
Blobs [Исходник н...
32 урока по Delphi
Форма в форме
C# в кратком изло...
ActiveX в Delphi
Как программирова...
Книга по Delphi (...
PDF
AID антивирус
JanReplace
Tenis [Исходник н...
Сапёр
Правила программи...
С. Г. Горнаков - ...

Топ загрузок
Приложение Клие... 100801
Delphi 7 Enterp... 98068
Converter AMR<-... 20304
GPSS World Stud... 17069
Borland C++Buil... 14262
Borland Delphi ... 10389
Turbo Pascal fo... 7399
Калькулятор [Ис... 6094
Visual Studio 2... 5242
Microsoft SQL S... 3677
Случайные статьи
Еще одна программа...
Динамические VLAN....
Слоты Вулкана
Карта состояния па...
Функция GetBkColor...
Имитация файлов и ...
Присвоение начальн...
Модальная скорость
Сульшер может лиши...
Как переносить уче...
Казино Parimatch
Влияние ЭВМ на сис...
Atari 5200 SuperSy...
Спец. команды
Задача 3 посвящена...
Преобразование теп...
• Удаление одинако...
Создание уникально...
Точная арифметика
Задача реализации ...
Небольшие изменени...
Идеальное решение
Выполнение агрегир...
Как отлично провес...
Фаза подтверждения
Статистика



Друзья сайта
Программы, игры


Полезно
В какую объединенную сеть входит классовая сеть? Суммирование маршрутов Занимают ли таблицы память маршрутизатора?