C++:派生类的构造函数和析构函数

简介:

                        4.2 派生类的构造函数和析构函数
4.2.1 派生类构造函数和析构函数的执行顺序
通常情况下,当创建派生类对象时,首先执行基类的构造函数,随后再执行派生类的构造函数;
当撤销派生类对象时,则先执行派生类的派生类的析构函数,随后再执行基类的析构函数。 

//例4.5 派生类的构造函数和析构函的执行顺序

复制代码
#include<iostream>
using namespace std;
class Base{         //声明基类Base 
  public:
    Base()
    {
      cout<<"Constructor Base Class.."<<endl;
    }
    ~Base()
    {
      cout<<"Destructor Base Class.."<<endl;
    }
};
class Derived: public Base{     //声明基类Base的公有派生类Derived 
  public:
    Derived()
    {
      cout<<"Constructor Derived Class.."<<endl;
    }
    ~Derived()
    {
      cout<<"Destructor Derived Class.."<<endl;
    }  
}; 
int main()
{
  Derived obj;
  return 0;
}
/*
 说明:从程序运行的结果可以看出:构造函数的调用严格按照先调用基类的构造函数,后调用派生类的
       构造函数的顺序执行。析构函数的调用顺序与构造函数的调用顺序正好相反,先调用派生类的析
       构函数,再调用基类的析构函数。 
*/
复制代码

4.2.2 派生类构造函数和析构函数的构造规则

1. 简单的派生类的构造函数

当基类的构造函数没有参数,或没有显式定义构造函数时,派生类可以不向基类传递参数,甚至可以不定义构造函数。例4.5的程序就是由于基类的构造函数没有参数,所以派生类没有向基类传递参数。

派生类不能继承基类中构造函数和析构函数。当基类含有带参数的构造函数时,派生类必须定义构造函数,以提供把参数传递给基类构造函数的途径。

在C++中,派生类构造函数的一般格式为:

派生类名(参数总表):基类名(参数表)

派生类新增数据成员的初始化语句 
} 

其中,基类构造函数的参数,通常来源于派生类构造函数的参数总表,也可以用常数值。

//例4.6 当基类含有带参数的构造函数,派生类构造函数的构造方法。

复制代码
#include<iostream>
#include<string> 
using namespace std;
class Student{           //声明基类Student 
    public:
      Student(int number1,string name1,float score1) //基类的构造函数 
      {
        number = number1;
        name = name1;
        score = score1; 
      }
      void print()
      {
        cout<<"number:"<<number<<endl;
        cout<<"name:"<<name<<endl;
        cout<<"score:"<<score<<endl;
      }
    protected:
      int number;
      string name;
      float score; 
};
class UStudent:public Student{     //基类Student的公有派生类UStudent 
   public: 
     UStudent(int number1,string name1,float score1,string major1)
     :Student(22117,"张三",100/*number1,name1,score1*/)
     {                           //定义派生类的构造函数时,缀上要调用的基类的构造函数及其参数 
       major = major1;           //参数可以是派生类构造函数总参数表中的参数,也可以是常量和全局变量。
     }
     void print1()
     {
       print();
       cout<<"major:"<<major<<endl;
     }
   private:
     string major;      
};
int main()
{
 UStudent stu(22116,"张志",95,"信息安全");
 //stu.print();
 stu.print1();
 return 0;
} 
复制代码

注意事项:

请注意派生类构造函数首行的写法:
UStudent(int number1,string name1,float score1,string major1):Student(number1,name1,score1)

冒号前面的部分是派生类构造函数的主干,它和以前介绍过的构造函数的形式相同,但它的总参数表中包括基类构造函数所需要的参数和对派生类新增的数据成员初始化所需要的参数。冒号后面的部分是要调用的基类构造函数及其参数。

从上面列出的派生类UStudent构造函数首行中可以看到,派生类构造函数名(UStudent)后面的总参数表中包括参数的类型和参数名(如 int number1),而基类构造函数的参数表中只有参数名而不包括参数类型(如 number1),因为在这里不是再定义基类构造函数,而是调用基类构造函数,因此这些参数是实参而不是形参。它们可以是派生类构造函数总参数表中的参数,也可以是常量和全局变量。

说明:1、可以将派生类构造函数定义在类的外部,而在类体内只写该函数的声明。如在例4.6的派生类中可以只写构造函数的声明:
UStudent(int number1,string name1,float score1,string major1) 
而在类外定义派生类的构造函数:
UStudent::UStudent(int number1,string name1,float score1,string major1):Student(number1,name1,score1)
{
          major = major1;

请注意:在类中声明派生类构造函数时,不包括基类构造函数及其参数表(Student(number1,name1,score1),只在
类外定义构造函数时才将它列出。

2、若基类使用默认构造函数或不带参数的构造函数,则在派生类中定义构造函数时, "基类构造函数名(参数表)"可以省略。 
如在例4.5的程序中,由于基类的构造函数没有参数,所以在派生类中定义构造函数时,不要缀上":Base()",也即
不必写成:

Deirved:Base()
{
cout<<"Constructor derived class"<<endl;
}

3、当基类构造函数不带参数时,派生类不一定需要定义构造函数,然而当基类的构造函数哪怕只带有一个参数,它所有的派生类必须定义构造函数,甚至所定义的派生类构造函数的函数体可能为空,仅仅起到参数的传递作用。
例如,在下面的程序中,派生类Deruved就不使用参数n,n只是被传递给基类构造函数Base。

复制代码
 class Base{
                public:
                  Base(int n)
                  {
                    i = n;
                  }
                  void show()
                  {
                    cout<<"i="<<i<<endl;
                  }
                private:
                    int i;  
             };
             class Derived public Base{
                public:
                  Derived(int n):Base(n)   //构造函数参数表中只有一个参数, 
                  { }                      //传递给了要调用的基类构造函数Base 
             };                            //派生类构造函数体为空
复制代码

2.派生类的析构函数

在派生类中可以根据需要定义自己的析构函数,用来对派生类中的所增加的成员进行清理工作。基类的清理工作仍然有基类的析构函数负责。由于析构函数是不带参数的,在派生类中是否要定义析构函数与它所属基类的析构函数无关。在执行派生类的析构函数时,系统会自动调用基类的析构函数,对基类的对象进行清理。析构函数的调用顺序与构造函数正好相反,先执行派生类的析构函数,再执行基类的析构函数。


//例4.7 简单派生类的构造函数和析构函数的执行顺序。

复制代码
#include<iostream>
using namespace std;
class A{                                    //声明基类A 
  public:
     A()
     {
       cout<<"Constuctor class A"<<endl;   //基类的构造函数 
     }
     ~A()
     {
       cout<<"Destuctor class A"<<endl;    //基类的构造函数 
     }
}; 
class B:public A{
  public:
     B()
     {
       cout<<"Constuctor class B"<<endl;   //派生类的构造函数 
     }
     ~B()
     {
       cout<<"Destuctor class B"<<endl;    //派生类的构造函数 
     }
};
int main()
{
  B b;
  return 0;
}
/*
  运行结果如下:  Constuctor class A
                  Constuctor class B
                  Destuctor class B
                  Destuctor class A
*/
复制代码

3.含有对象成员(子对象)的派生类的构造函数

当派生类中含有内嵌的对象成员(也称子对象)时,其构造函数的一般格式为:

派生类(参数总表):基类名(参数表 0):对象成员1(参数表 1),......,对象成员n(参数表 n)
{
派生类新增成员的初始化语句 
}

在定义派生类对象时,构造函数的执行顺序如下:
(1)、调用基类的构造函数,对基类数据成员初始化;
(2)、调用内嵌对象成员的构造函数,对内嵌对象成员的数据成员进行初始化;
(3)、调用派生类的构造函数体,对派生类数据成员进行初始化。

撤销对象时,析构函数的调用顺序与构造函数的调用顺序相反,首先执行派生类的析构函数,
再执行内嵌对象成员的析构函数,最后执行基类的析构函数。 

//例4.8 含有对象成员的派生类构造函数和析构函数的执行顺序

复制代码
#include<iostream>
using namespace std;
class Base{                   //声明基类Base 
  public:
    Base(int i)               //基类的构造函数 
    {
     x = i;
     cout<<"Constructor base class"<<endl;
    }
    ~Base()                   //基类的析构函数 
    {
     cout<<"Destructor base class"<<endl;
    }
    void show()
    {
     cout<<"x = "<<x<<endl;
    }
    private:
      int x;
}; 
class Derived :public Base{     //声明基类Base的公有派生类Derived 
    public:
    Derived(int i):Base(i),d(i) //派生类的构造函数,缀上要调用的基类构造函数和对象成员的构造函数
    {
      cout<<"Constructor Derived class"<<endl;
    }
    ~Derived()
    {
      cout<<"Destructor Derived class"<<endl;
    } 
    private:
    Base d;
}; 
int main()
{
 Derived obj(5);
 obj.show();
 return 0;
}
/*
 运行结果是:Constructor base class
             Constructor base class
             Constructor Derived class
             x = 5
             Destructor Derived class
             Destructor base class
             Destructor base class
                           
*/
复制代码

说明:
(1)在派生类中含有多个内嵌对象成员时,调用内嵌对象成员的构造函数顺序由它们在类中声明的顺序确定。
(2)如果派生类的基类也是一个派生类,每个派生类只需负责其直接基类数据成员。依次上溯。 

//例4.9 含有多个对象成员的派生类构造函数的执行顺序。

复制代码
#include<iostream>
#include<string>
using namespace std;
class Student{            //声明基类Base 
  public:
    Student(int number1,string name1,float score1)
    {
      number = number1;
      name = name1;
      score = score1;
    }
    void print()
    {
      cout<<"学号:"<<number<<endl;
      cout<<"姓名:"<<name<<endl;
      cout<<"成绩:"<<score<<endl;
    }
  protected:
    int number;
    string name;
    float score;
};
class UStudent:public Student{      //声明基类Student的公有派生类UStudent 
  public:
    UStudent(int number1,string name1,float score1,int number2,string name2,float score2,
    int number3,string name3,float score3,string major1):Student(number1,name1,score1),
    auditor2(number3,name3,score3),auditor1(number2,name2,score2)  
    {
     major = major1;
    }
    void print()
    {
     cout<<"正式生是:"<<endl;    
     Student::print();
     cout<<"专业是:"<<major<<endl;
    }
    void print_auditor1()
    {
     cout<<"旁听生是:"<<endl;
     auditor1.print(); 
    }
    void print_auditor2()
    {
     cout<<"旁听生是:"<<endl;
     auditor2.print(); 
    }
  private:
    string major;                 //专业 
    Student auditor1;             //定义对象成员1(旁听生) 
    Student auditor2;             //定义对象成员2(旁听生)
};
int main()
{
 UStudent stu(2001,"张志",95,3001,"王全安",66,3002,"李倩倩",50,"信息安全");
 stu.print();
 
 stu.print_auditor1();
 stu.print_auditor2();
 return 0;
} 
复制代码

 

程序猿神奇的手,每时每刻,这双手都在改变着世界的交互方式!
本文转自当天真遇到现实博客园博客,原文链接:http://www.cnblogs.com/XYQ-208910/p/4912347.html ,如需转载请自行联系原作者
相关文章
|
5月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
158 0
|
5月前
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
249 0
|
7月前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
293 12
|
8月前
|
设计模式 安全 C++
【C++进阶】特殊类设计 && 单例模式
通过对特殊类设计和单例模式的深入探讨,我们可以更好地设计和实现复杂的C++程序。特殊类设计提高了代码的安全性和可维护性,而单例模式则确保类的唯一实例性和全局访问性。理解并掌握这些高级设计技巧,对于提升C++编程水平至关重要。
174 16
|
8月前
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
|
8月前
|
存储 编译器 C++
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。
|
8月前
|
编译器 C++
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
|
9月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
9月前
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)
|
8月前
|
安全 C++
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
476 6