Г. Шилтд - Самоучитель C++ (PDF) (1114887), страница 32
Текст из файла (страница 32)
Синтаксис передачи аргументов из производного в базовый класс показан ниже:конструктор_произв_класса(список-арг) : базов_класс (список_арг)// тело конструктора производного класса{Для базового и производного классов допустимо использовать одни и те жеаргументы. Кроме этого, для производного класса допустимо игнорированиевсех аргументов и передача их напрямую в базовый класс.1. В этой очень короткой программе показано, в каком порядке выполняютсяконструкторы и деструкторы базового и производного классов:^include <iostream>using namespace std;.class base {public:b a s e ( ) { cout « "Работа конструктора базового класса \п"; }~base ( ) { cout « "Работа деструктора базового класса \п"; }class derived: public base (public:derived() ( cout « "Работа конструктора производного класса\п"; }-derived () ( cout « "Работа деструктора производного классаХп"; }int main (){derived о;return 0;После выполнения программы на экран выводится следующее:Работа конструктора базового классаРабота конструктора производного классаРабота деструктора производного классаРабота деструктора базового классаКак видите, конструкторы выполняются в порядке наследования, а деструкторы — в обратном порядке.218______Самоучитель2.
В этой программе показана передача аргумента конструктору производногокласса:linclude <:iostream>using namespace std;class base {public:base() { cout « "Работа конструктора базового класса\п"; }-base () { cout « "Работа деструктора базового класса\п"; }class derived: public base {int j ;public:derived(int n) {cout « "Работа конструктора производного класса \п";j = n;}~derived{) { cout « "Работа деструктора производного класса\п"; }void showj() { cout « j « '\n'; }};int main()/derived о (10) ;о . showj ( ) ;return 0;Обратите внимание, что аргумент передается конструктору производногокласса обычным образом.3.
В следующем примере у конструкторов производного и базового классовимеются аргументы. В этом особом случае оба конструктора используютодин и тот же аргумент, и производный класс просто передает этот аргументв базовый класс.#include <iostream>using namespace std;class base {int i;public:base {int n) {cout « "Работа конструктора базового класса\п";i - n;Глава 7. Наследование2)9-base () { cout « "Работа деструктора базового класса\п"; }void show!() ( cout « i « ' \ n ' ; }class derived: public base {int j ;public:derived{int n} : base (n) { // передача аргумента/ / в базовый классcout « "Работа конструктора производного классаЧп";j = n;}~derived() { cout « "Работа деструктора производного класса\п)void showj() { cout « j « '\n'; });int main ()tderived о (10) ;o.showi ( ) ;o.showj () ;return 0;Обратите особое внимание на объявление конструктора производного класса.Отметьте, как параметр n (который получает аргумент при инициализации)используется в конструкторе derived() и передается конструктору base().4.
Обычно конструкторы базового и производного классов не используют одини тот же аргумент. В этом случае, при необходимости передать каждому конструктору класса один или несколько аргументов, вы должны передать конструктору производного класса все аргументы, необходимые конструкторамобоих классов. Затем конструктор производного класса просто передает конструктору базового класса те аргументы, которые ему требуются. Например,в представленной ниже программе показано, как передать один аргументконструктору производного класса, а другой — - конструктору базового класса:#include <iostream>using namespace std;class base {int- i;public :base (int n) {cout « "Работа конструктора базового класса\п" ;i = n;\220Самоучитель C++-base() { cout « "Работа деструктора базового класса\п"; )void showi() { cout « i « '\n'; }class derived: public base {int j ;public:derivedfint n, int m) : base (m) { // передача аргумента//в базовый классcout « "Работа конструктора производного класса\п";j = п;}~derived() { cout « "Работа деструктора производного класса\п}void showj() { cout « j « '\n'; }};int main(){derived о (10, 20) ;о.
showi () ;o.showj () ;return 0;5. Конструктору производного класса совершенно нет необходимости как-тообрабатывать аргумент, предназначенный для передачи в базовый класс. Если производному классу этот аргумент не нужен, он его просто игнорирует ипередает в базовый класс. Например, в этом фрагменте параметр п конструктором derived() не используется. Вместо этого он просто передается конструктору base():class base {int i;public:base (int n) {cout « "Работа конструктора базового класса\п";i = п;}-base () { cout « "Работа деструктора базового класса\п"; }void showi () ( cout « i « *\n'; }class derived: public base {int j ;public:derived(int n) : base(n) { // передача аргумента в базовый классГлава 7. Наследование221cout « "Работа конструктора производного класса\п";j = 0; // аргумент п здесь не используется})-derived() { cout « "Работа деструктора производного класса\п";void showj(} { cout « j « '\n'; }[Упражнения]&^1.
В приведенном ниже фрагменте добавьте конструктор для класса myderived.Он должен передать указатель на инициализируемую строку конструкторукласса mybase. Кроме того, конструктор myderived() должен инициализировать переменную len длиной строки.ttinclude <iostream># include <cstring>using namespace std;class rnybase {char str[80] ;public :mybasefchar *s) { strcpy(str, s) ; }char * g e t ( ) { return str; }class myderived: public mybase {int len;public:// добавьте здесь конструктор myderived ()int getlenf) ( return len; }void showf) ( cout « get() « '\n'; }};int main (){myderived ob ("привет") ;ob . show ( ) ;cout « ob.getlen() « '\n';return 0;2.
Используя следующий фрагмент, создайте соответствующие конструкторысаг() и truck(). Они должны передавать необходимые аргументы объектамкласса vehicle. Кроме этого конструктор саг() должен при создании объекта222Самоучитель C++инициализировать переменную passengers, а конструктор truck() — переменную loadlimit.^include <iostream>using namespace std;// Базовый класс автомобилей для разных типовclass vehicle {int num_wheels;int range;public:vehicle(int w, int r)num_wheels = w; range = r;void showv(),cout « "Число колес" « num_wheels « '\n';cout « "Грузоподъемность" « range « '\n';J;class car: public vehicle {int passengers;public:// вставьте здесь конструктор саг()void show()showv();cout « "Пассажироемкость" « passengers « '\n';class truck: public vehicle {int loadlimit;public:// вставьте здесь конструктор truck {)void ahow(){showv {) ;cout « "Пробег" « loadlimit « ' \n ' ;int main ()(car c(5, 4, 500) ;truck t (30000, 12, 1200);Глава\7._ Наследование223cout « "легковая машина:\п";с.show();cout « "Хпгрузовик:\п";t.
show () ;return 0;Для конструкторов саг() и truck() объекты должны объявляться следующимобразом:car ob(passengers, wheels, range);truck ob(loadlimit, wheels, range);7.4. Множественное наследованиеИмеются два способа, посредством которых производный класс может наследовать более одного базового класса. Во-первых, производный класс может использоваться в качестве базового для другого производного класса,создавая многоуровневую иерархию классов.
В этом случае говорят, что исходный базовый класс является косвенным (indirect) базовым классом длявторого производного класса. (Отметьте, что любой класс — независимо оттого, как он создан — может использоваться в качестве базового класса.)Во-вторых, производный класс может прямо наследовать более одного базового класса. В такой ситуации созданию производного класса помогает комбинация двух или более базовых классов. Ниже исследуются результаты, ккоторым приводит наследование нескольких базовых классов.Когда класс используется как базовый для производного, который, в своюочередь, является базовым для другого производного класса, конструкторывсех трех классов вызываются в порядке наследования.
(Это положение является обобщением ранее исследованного принципа.) Деструкторы вызываются в обратном порядке. Таким образом, если класс В1 наследуетсяклассом Dl, a D1 — классом D2, то конструктор класса В1 вызывается первым, за ним конструктор класса D1, за которым, в свою очередь, конструктор класса D2. Деструкторы вызываются в обратном порядке.Если производный класс напрямую наследует несколько базовых классов,используется такое расширенное объявление:class имя_проязводного_кла.сса:сп_доступа кмя_базового_класса1,сп доступа лмя_баэовог^> класса.2,. . . , сп доступа иия_базовосо-классам{I I . .
.тело класса224_СамоучительC++Здесь цмя_базового_класса1 ... UMR_6a3oeozo_tuiaccaN— имена базовых классов, сп_доступа — спецификатор доступа, который может быть разным уразных базовых классов. Когда наследуется несколько базовых классов,конструкторы выполняются слева направо в том порядке, который задан вобъявлении производного класса. Деструкторы выполняются в обратномпорядке.Когда класс наследует несколько базовых классов, конструкторам которыхнеобходимы аргументы, производный класс передает эти аргументы, используя расширенную форму объявления конструктора производного класса:констр произв_класса (список_арг) : имя базового класса! (список арг) ,имя_базового_хласса2 (список_арг) ,.
. . , имя базового_классаК( список{// ... тело конструктора производного классаЗдесь имя_бозового_класса! ... uMR_6o3oeoto_KnaccaN — имена базовых классов.Если производный класс наследует иерархию классов, каждый производныйкласс должен передавать предшествующему в цепочке базовому классу всенеобходимые аргументы.Примеры1.
В этом примере производный класс наследует класс, производный от другогокласса. Обратите внимание, как аргументы передаются по цепочке от классаD2 к классу В1.// Множественное наследование^include <iostream>using namespace std;class Bl {int a;public:B l ( i n t x) { a = x; }int geta!) { return a; }// Прямое наследование базового классаclass Dl: public Bl {int b;public:Dl(int x, int y): Bl(у) // передача переменной у классу Bl225Глава 7, Наследование(11Ь = х;int getbO t return b; }// Прямое наследование производного класса//и косвенное наследование базового классаclass D2: public Dl {int с;public:D2 (int x, int y, int z) : Dl{y, z) // передача аргументов// классу Dl{с = x;/* Поскольку базовые классы наследуются как открытые, класс D2 имеетдоступ к открытым элементам классов В1 и Dl */void show ( ) {cout « geta() « ' ' « getb() « ' ';cout « с « ' \n ' ;int main ( ){D2 ob(l, 2, 3);ob.showf) ;// функции geta() и getb ( ) здесь тоже открытыcout « ob.geta{) « ' ' « ob.getb{) « '\n';return 0;Вызов функции ob.showQ выводит на экран значения 3 2 1.