cpp-oop (823968), страница 8
Текст из файла (страница 8)
«Объектно-ориентированное программирование на языке C++ в среде Microsoft Visual Studio 2008»51cout<<"y = "<<y<<endl;cout<<"z = "<<z<<endl;}};void main(){A aa; // создание объекта базового классаB bb; // создание объекта производного классаaa.print();// выводит: x = 20 y = 30 z = 50bb.print();// выводит: y = 30 z = 50system("pause");}ОглавлениеИванова Г.С., Ничушкина Т.Н. «Объектно-ориентированное программирование на языке C++ в среде Microsoft Visual Studio 2008»522.2Множественное наследованиеЯзык С++ позволяет осуществлять наследование не только от одного, но иодновременноотнесколькихклассов.Такоенаследованиеполучилоназваниемножественного. Описание производного класса при множественном наследованиивыглядит следующим образом:class <Имя производного класса>:<Вид наследования><Имя базового класса 1>,<Вид наследования><Имя базового класса 2>,...<Вид наследования><Имя базового класса n> {...};Вид наследования, как и в случае простого наследования, определяет режим доступак компонентам соответствующего базового класса.
Если конструкторы базовых классов неимеют аргументов, то производный класс может не иметь конструктора. При наличии уконструкторов базового класса аргументов производный класс обязан иметь конструкторсо списком инициализации следующего вида:<Имя конструктора производного класса>(<Список аргументов>):<Имя конструктора базового класса 1>(<Список аргументов 1>),......<Имя конструктора базового класса n>(<Список аргументов n>){<Тело конструктора производного класса>}Последовательность активизации конструкторов такая же, как и для случаяединственного базового класса: сначала активизируются конструкторы всех базовыхклассов в порядке их перечисления в объявлении производного класса, затемконструкторы объектных полей и в завершении – конструктор производного класса.Соответственно поля базовых классов создаются в том порядке, в котором эти поляперечислены при объявлении производного класса.ОглавлениеИванова Г.С., Ничушкина Т.Н.
«Объектно-ориентированное программирование на языке C++ в среде Microsoft Visual Studio 2008»53Пример 2.4. Наследование от двух базовых классов. В данном примере класс thirdнаследуется от классов first и second (рис. 2.2).secondfirstnum_fc_s, num_sfirst()second()thirdtwonasl()get()Рис. 2.2. Множественное наследованиеОбъект класса third состоит из следующих полей:• поля num_f, унаследованного от класса first (описанного public,наследованного private, следовательно доступного private) и инициализированногослучайным числом в диапазоне от -50 до 49;• полей num_s и c_s, унаследованных от класса second (описанных public,наследованных public, следовательно доступного public) и инициализированныхчислами 20 и символом «R», причем инициализация поля c_s в конструкторе классаsecond не предусмотрена, поэтому она выполняется в теле конструктора класса third.#include <locale.h>#include <iostream>#include <conio.h>#include <stdlib.h>#include<time.h>using namespace std;class first{public:int num_f;first(int va):num_f(va){cout<<"Конструктор first\n";}};class second{ОглавлениеИванова Г.С., Ничушкина Т.Н.
«Объектно-ориентированное программирование на языке C++ в среде Microsoft Visual Studio 2008»54public:char c_s;int num_s;second(int vn):num_s(vn){cout<<"Конструктор second\n"; }};class third:private first, // сокрытие базового класса// открытый базовый классpublic second{public:third(int nm,char vc,int nfx):first(nfx), second(nm){cout<<"Конструктор third\n";c_s=vc; /* инициализация поля базового классаконструктором производного класса */}int get(void){return num_f;// получение значения внутреннего поля}};void main(){setlocale(0,"russian");srand((unsigned)time(NULL)); // инициализация датчика случайных чиселint r=rand()/1000-rand()/1000; // присвоение числу случайного значенияthird aa(r,'R',50);cout<<aa.get()<<" "<<aa.c_s<<" "<<aa.num_s<<endl;_getch();}В результате выполнения программы мы получаем следующую цепочку обращенийк конструкторам:Конструктор firstКонструктор secondКонструктор thirdОглавлениеИванова Г.С., Ничушкина Т.Н.
«Объектно-ориентированное программирование на языке C++ в среде Microsoft Visual Studio 2008»55и содержимое полей num_f, c_s и num_s, доступных в классе third:50 R -4ОглавлениеИванова Г.С., Ничушкина Т.Н. «Объектно-ориентированное программирование на языке C++ в среде Microsoft Visual Studio 2008»562.3Виртуальное наследованиеПри множественном наследовании базовый класс не может быть указан впроизводном классе более одного раза.
Однако возможна ситуация, когда производныйкласс при наследовании от потомков одного базового класса многократно наследует однии те же компоненты базового класса (рис. 2.3). Иными словами, производный класс будетсодержать несколько копий полей одного базового класса.Base..................DescentРис. 2.3. Многократное наследование полей базового классаЧтобы избежать многократного включения в производный класс компонентовбазового класса, используется виртуальное наследование. При виртуальном наследованиипроизводный класс описывают следующим образом:class <Имя производного класса>:virtual <Вид наследования><Имя базового класса>{...};Вэтом случае включение в производныйкласс полейбазовогоклассаосуществляется один раз, а их инициализация выполняется в производном классе,который не является прямым потомком базового класса. Вызов конструкторов при этомпроисходит в следующем порядке: сначала конструктор виртуально наследуемогобазового класса, затем конструкторы базовых классов в порядке их перечисления приобъявлении производного класса, за ними – конструкторы объектных полей и конструкторпроизводного класса.
Деструкторы соответственно вызываются в обратном порядке.Виртуально наследуемый класс обязательно должен содержать конструктор безпараметров, который активизируется при выполнении конструкторов классов – прямыхпотомков виртуально наследуемого класса.ОглавлениеИванова Г.С., Ничушкина Т.Н. «Объектно-ориентированное программирование на языке C++ в среде Microsoft Visual Studio 2008»57Пример2.5.Виртуальноенаследование.Реализуемиерархиюклассов,представленную на рис. 2.3. Класс derived наследуется от двух наследников класса fixed.Чтобы исключить удваивание полей, описанных в классе fixed, необходимо использоватьвиртуальное наследование.#include <locale.h>#include <iostream>using namespace std;class fixed{protected:int Fix;public:fixed() // конструктор без параметров{cout<<" Конструктор класса fixed\n";}fixed(int{fix):Fix(fix)// конструктор c параметромcout<<" Конструктор класса fixed int\n"; }};class derived_1: virtual public fixed // виртуальное наследование{public:int One;derived_1(void)cout<<" Конструктор класса derived 1\n"; }{};class derived_2: virtual private fixed // виртуальное наследование{public:int Two;derived_2(void){cout<<" Конструктор класса derived 2\n";}};class derived: public derived_1, public derived_2/* объявление производного класса – непрямого потомка */{public:derived(void)ОглавлениеИванова Г.С., Ничушкина Т.Н.
«Объектно-ориентированное программирование на языке C++ в среде Microsoft Visual Studio 2008»58{ cout<<" Конструктор класса derived() \n";}derived(int fix):fixed(fix){ cout<<" Конструктор класса derived (int) \n";}void Out( ){ cout<<" Значение поляFix = "<< Fix<<endl;}};void main(){setlocale(0,"russian");derived Var(10);Var.Out( );system("pause");}В результате работы программы получаемКонструктор класса fixedintКонструктор класса 1Конструктор класса 2Конструктор класса derived (int)Значение поля Fix=10Если бы наследование не было виртуальным, поле Fix было бы включено в объекткласса derived дважды:derived_1::Fixиderived_2::Fix.ОглавлениеИванова Г.С., Ничушкина Т.Н.
«Объектно-ориентированное программирование на языке C++ в среде Microsoft Visual Studio 2008»592.4Простой полиморфизмВ языке С++ предусмотрен м е х а н и з мп о л и м о р ф и з м а, обеспечивающийвозможность определения разных описаний некоторого единого по названию метода дляклассов различных уровней иерархии.
При этом различают простой полиморфизм,базирующийсянамеханизмераннегосвязывания,исложныйполиморфизм,использующий механизм позднего связывания.Простой (можно использовать термин "статический") полиморфизм поддерживаетсяязыком С++ на этапе компиляции и реализуется с помощью механизма переопределения(перегрузки) функций. Поэтому такие полиморфные функции называются в С++переопределяемыми. В соответствии с общими правилами переопределения функций онидолжны отличаться сигнатурой, т.
е. количеством, типом и порядком следованияпередаваемых параметров.Подобный подход позволяет строить более гибкие и совершенные иерархии классов,переопределяя в производных классах методы в соответствии с требованиямиразрабатываемой программы или системы, использующей эти классы.Пример 2.6. Использование раннего связывания. Для демонстрации применениястатического полиморфизма вернемся к примеру 2.2. В этом примере функции print()и printr() выполняют одинаковые по смыслу операции – выводят главные поляобъекта на экран, следовательно, их можно назвать одним именем, например print().Метод print(), таким образом, станет полиморфным – переопределяемым впроизводном классе. Отдельное определение метода в своем классе называют аспектомполиморфного метода.#include <locale.h>#include <stdio.h>#include <stdlib.h>#include <iostream>#include <string.h>using namespace std;typedef unsigned long dlong;class Tlong// Класс Целое числоОглавлениеИванова Г.С., Ничушкина Т.Н.
«Объектно-ориентированное программирование на языке C++ в среде Microsoft Visual Studio 2008»60{public:dlong num;// числовое поле классаTlong(){}// неинициализирующий конструкторTlong(dlong an):num(an){} // конструктор~Tlong(){}// деструкторvoid print(void)// вывод значения поля{ cout<<" Целое число : "<<num<<endl; }void setnum(dlong an) // инициализации поля{ num=an; }};class// Класс Вещественное числоTreal: public Tlong{public:dlong drob;// дробная часть числаchar *real;// запись вещественного числаTreal(){}// неинициализирующий конструкторTreal(char *st) :Tlong() // конструктор{setnumv(st); }// деструктор~Treal(){delete real;void print();}// вывод вещественного числа (переопределяется)void setnumv(char * st);// инициализация полей класса};void Treal::setnumv(char * st){int l;char *ptr;l=strlen(st); real=new char[l+1]; strcpy(real,st);ptr=strchr(real,'.'); *ptr='\0';drob=dlong(atol(ptr+1));num=dlong(atol(real));*ptr='.';}ОглавлениеИванова Г.С., Ничушкина Т.Н.