C++ за 21 день седьмое издание



Download 1,38 Mb.
bet193/437
Sana22.02.2022
Hajmi1,38 Mb.
#89455
TuriРеферат
1   ...   189   190   191   192   193   194   195   196   ...   437
Bog'liq
word1

#include




  1. using namespace std;








  1. class Fish




  1. {




  1. public:




  1. virtual void Swim()




  1. {

  2. cout « "Fish swims!" « endl;




  1. }

10: }; /
1 1 :



  1. class Tuna:public Fish




  1. {




  1. public:




  1. // переопределение Fish::Swim




  1. void Swim()




  1. {




  1. cout « "Tuna swims!" « endl;




  1. }

20: };
2 1 :



  1. class Carp:public Fish




  1. {




  1. public:

  2. // переопределение Fish::Swim




  1. void Swim()




  1. {




  1. cout « "Carp swims!" « endl;




  1. }




  1. };



31:

Основы полиморфизма

281




  1. void MakeFishSwim(Fish& InputFish)




  1. {




  1. // вызов виртуального метода Swim() InputFish.Swim();

  2. }








  1. int main()

59 {

  1. Tuna myDinner;




  1. Carp myLunch;








  1. // передача Tuna как Fish




  1. MakeFishSwim(myDinner);








  1. // передача Carp как Fish

  2. MakeFishSwim(myLunch);








  1. return 0;




  1. }



Результат


Tuna swims!


Carp swims!


Анализ

Реализация функции M ak eF ish S w im (F ish & ) никак не изменилась с листинга 11.1,





  1. вывод получился совсем иной. Метод F i s h : : Swim () не был вызван вообще из-за при­ сутствия переопределенных версий T u n a : : Swim () и C a r p : : Swim (), которые получили преимущество над методом F i s h : : Swim (), поскольку последний был объявлен как вир­ туальная функция. Это очень важный момент. Он свидетельствует, что, даже не зная точ­ ный тип обрабатываемого объекта, класс которого происходит от класса F is h , реализация метода M akeFishSw im () способна привести к вызову разных реализаций метода Swim (), определенного в различных производных классах.

Это полиморфизм: обработка различных рыб как общего типа F is h , при гарантии выполнения правильной реализации метода Swim (), предоставляемого производными типами.




Потребность в виртуальных деструкторах



  1. средств, представленных в листинге 11.1, есть и оборотная сторона: неумышленный вызов функций базового класса из экземпляра производного, когда доступна специализа­ ция. Что будет, если функция применит оператор d e l e te , используя указатель типа B ase*, который фактически указывает на экземпляр производного класса?

Какой деструктор будет вызван? Рассмотрим листинг 11.3.


ЛИСТИНГ 1 1 .3 . Функция, вызываю щ ая оператор delete для типа Base*





  1. #include




  1. using namespace std;



2 :

282 ЗАНЯТИЕ 11. Полиморфизм





  1. class Fish




  1. {




  1. public:




  1. Fish()




  1. {




  1. cout « "Constructed Fish" « endl;




  1. }




  1. ~Fish()



11: {

  1. cout « "Destroyed Fish" « endl;




  1. }




  1. };



15:



  1. class Tuna:public Fish




  1. {




  1. public:




  1. TunaO

20: {



  1. cout « "Constructed Tuna" « endl;




  1. }

  2. -TunaO




  1. {




  1. cout « "Destroyed Tuna" « endl;




  1. }




  1. };



28:



  1. void DeleteFishMemory(Fish* pFish)




  1. {




  1. delete pFish;




  1. }



33:



  1. int main()




  1. {




  1. cout « "Allocating a Tuna on the free store:" « endl;




  1. Tuna* pTuna = new Tuna;




  1. cout « "Deleting the Tuna: " « endl;




  1. DeleteFishMemory(pTuna);



40:



  1. cout « "Instantiating a Tuna on the stack:" « endl;




  1. Tuna myDinner;




  1. cout « "Automatic destruction as it goes out of scope:




  1. endl;



44:



  1. return 0;




  1. }



Результат


Allocating a Tuna on the free store:


Constructed Fish


Constructed Tuna


Deleting the Tuna:


Destroyed Fish


Instantiating a Tuna on the stack:

Основы полиморфизма

283



Constructed Fish


Constructed Tuna


Automatic destruction as it goes out of scope:


Destroyed Tuna


Destroyed Fish


Анализ

Функция m ain () создает экземпляр класса Tuna в динамической памяти, используя оператор new в строке 37, а затем сразу освобождает зарезервированную память, исполь­ зуя вспомогательную функцию D eleteF ish M em o ry () в строке 39. Для сравнения: другой экземпляр класса Tuna создается в стеке как локальная переменная m yD inner (строка 42)



    1. выходит из области видимости по завершении функции m ain (). Вывод создают опера­ торы c o u t в конструкторах и деструкторах классов F is h и Tuna. Обратите внимание: не­ смотря на то, что обе части, Tuna и F is h , объекта были созданы в динамической памяти, поскольку использовался оператор new, при удалении был вызван только деструктор части F ish , а не класса Tuna. Это находится в абсолютном контрасте с созданием и удалением локального объекта m yD inner, где вызываются все конструкторы и деструкторы . В ли ­ стинге 10.7 был представлен правильный порядок создания и удаления классов в иерархии наследования, демонстрирующий, что должны быть вызваны все деструкторы, включая деструктор -T u n a (). Здесь явно что-то неправильно.

Проблема в том, что код в деструкторе производного класса, объект которого был соз­ дан в динамической памяти при помощи оператора new, не будет вызван, если будет при­ менен оператор d e l e t e для указателя типа B ase* . В результате ресурсы не будут осво­ бождены, произойдет утечка памяти и другие ненужные неприятности.


Чтобы избежать этой проблемы, используйте виртуальные деструкторы, как показано





  1. листинге 11.4.

ЛИСТИНГ 1 1 .4 . Использование виртуальных деструкторов для гарантии вы зова деструкторов производных классов при удалении указателя типа Base*___________________





  1. #include




  1. using namespace std;

2 :



  1. class Fish




  1. {




  1. public:




  1. Fish()

■>: {


  1. cout « "Constructed Fish" « endl;




  1. }




  1. virtual -Fish() // виртуальный деструктор!




  1. {




  1. cout « "Destroyed Fish" « endl;




  1. }




  1. };



15:


15: class Tuna:public Fish

  1. {

  2. public:




  1. TunaO




  1. {

284 ЗАНЯТИЕ 11. Полиморфизм





  1. cout << "Constructed Tuna" « endl;



22:}



  1. ~Tuna()

24: {

  1. cout « "Destroyed Tuna" « endl;




  1. }




  1. };



28:



  1. void DeleteFishMemory(Fish* pFish)




  1. {




  1. delete pFish;




  1. }



33:



  1. int main()




  1. {




  1. cout « "Allocating a Tuna on the free store:" « e n d l ;




  1. Tuna* pTuna = new Tuna;




  1. cout « "Deleting the Tuna: " « endl;




  1. DeleteFishMemory(pTuna);



40:



  1. cout « "Instantiating a Tuna on the stack:" « endl;




  1. Tuna myDinner;




  1. cout « "Automatic destruction as it goes out of scope:




  1. endl;



44:



  1. return 0;




  1. }



Результат


Allocating a Tuna on the free store:


Constructed Fish


Constructed Tuna


Deleting the Tuna:


Destroyed Tuna


Destroyed Fish


Instantiating a Tuna on the stack:



Download 1,38 Mb.

Do'stlaringiz bilan baham:
1   ...   189   190   191   192   193   194   195   196   ...   437




Ma'lumotlar bazasi mualliflik huquqi bilan himoyalangan ©hozir.org 2024
ma'muriyatiga murojaat qiling

kiriting | ro'yxatdan o'tish
    Bosh sahifa
юртда тантана
Боғда битган
Бугун юртда
Эшитганлар жилманглар
Эшитмадим деманглар
битган бодомлар
Yangiariq tumani
qitish marakazi
Raqamli texnologiyalar
ilishida muhokamadan
tasdiqqa tavsiya
tavsiya etilgan
iqtisodiyot kafedrasi
steiermarkischen landesregierung
asarlaringizni yuboring
o'zingizning asarlaringizni
Iltimos faqat
faqat o'zingizning
steierm rkischen
landesregierung fachabteilung
rkischen landesregierung
hamshira loyihasi
loyihasi mavsum
faolyatining oqibatlari
asosiy adabiyotlar
fakulteti ahborot
ahborot havfsizligi
havfsizligi kafedrasi
fanidan bo’yicha
fakulteti iqtisodiyot
boshqaruv fakulteti
chiqarishda boshqaruv
ishlab chiqarishda
iqtisodiyot fakultet
multiservis tarmoqlari
fanidan asosiy
Uzbek fanidan
mavzulari potok
asosidagi multiservis
'aliyyil a'ziym
billahil 'aliyyil
illaa billahil
quvvata illaa
falah' deganida
Kompyuter savodxonligi
bo’yicha mustaqil
'alal falah'
Hayya 'alal
'alas soloh
Hayya 'alas
mavsum boyicha


yuklab olish