C++类的“包含”机制

来源:互联网 发布:不用网络的音乐播放器 编辑:程序博客网 时间:2024/06/04 17:42
//何为“包含”,其实说白了就是一个类可以包含另一个类的对象,即如下程序所示:
class A{                   //... }; class B{                   //...    A a;    A b;};

在上面这个程序中,我们定义了类A和类B。其中类B里面我们定义了类A的两个对象a和b。这样的情况就叫类B包含了类A。下面,我们用一个程序来看一下“包含”:

#include <iostream>using namespace std;class A{public:    A(int i){x=i;cout<<"调用A类的构造函数\n";}    ~A(){cout<<"调用A类的析构函数\n";};    void get() {cout<<"A类中X的值为:"<<x<<endl;}private:    int x;}; class B{public:    B(int i,int j,int k):a(i),b(j),y(k){cout<<"调用B类的构造函数\n";}    A geta(){return a;}    A getb(){return b;}    ~B(){cout<<"调用B类的析构函数\n";}    void gety(){cout<<"B类中y的值为:"<<y<<endl;}private:    A a;    A b;    int y;}; int main(){    B b(1,2,3);    b.geta().get();    b.getb().get();    b.gety();    return 0;}


首先是对两个类进行分析:在上面这个程序中,我们定义了两个类A和B。其中可以看到,在类B的私有成员变量里面,我们定义了一个类B自己的成员变量,另外还定义了两个类A的对象a和b(22行和23行);另外在类B的公有函数中,我们定义了两个返回值为类A的函数:geta()和getb(),它们的作用就是返回在类B中定义的两个类A的对象a和b。在这里我们特别应该注意的是类B的构造函数:

B(int i,int j,int k):a(i),b(j),y(k){cout<<"调用B类的构造函数\n";}


 

这个构造函数很有意思。我们可以看到它不仅初始化了自己的私有成员变量y,而且也顺带初始化了类A的两个对象a和b。那么它肯定会调用类A的构造函数,而且会调用两次。然后再调用一次B类自己的构造函数。那么析构的时候顺序应该就是相反的,首先调用B类的析构函数,然后再调用两次A类的析构函数。我们可以看到后面的程序输出图这样说滴,^_^。(见输出的红色框和黄色框)

  • 现在我们再来看一看主函数中的东东。首先在程序的第29行,我们定义了B类的对象b,并调用了A类和B类的构造函数初始化了类A的对象a、b和类B的对象b。然后在程序的第30行我们就可以看到在博文一开始介绍的Qt中的东东。这里我们就搞不懂了,它们到底是干啥用的,什么都不说了,先看一下运行结果:

          我们可以看到,返回的类A的对象a中x的值为1,另外一个类A的对象b中x的值为2。好了,豁然开朗了,我们来解释一下程序第30行和31行。第29行用类B的对象b来调用成员函数geta(),该函数是在17行定义的,作用就是返回类A的对象a,因此第30行

    b.geta().get();


    就相当于

    A a(1);a.get();

    这里的原因就是,因为b.geta()返回的是x的成员值为1的对象a,所以再调用类A的get()函数就可以省略了对象a了。那么同理,程序第31行就相当于

    A b(2);b.get();


    这就说明了类B可以通过成员函数来访问被包含的类A对象的成员变量。就是文章一开始提到的方式,其实它是隐藏了声明类A的对象,因为由于包含的原因,类B已经帮类A搞定了对象的声明了

  • 原创粉丝点击