本人在學(xué)習(xí)Qt的時(shí)候發(fā)現(xiàn)了一個(gè)非常有趣的現(xiàn)象。有很多函數(shù)的調(diào)用方法都寫(xiě)成了如下的形式:
這令小弟著實(shí)不懂。在上面這段代碼中,第一個(gè)對(duì)象調(diào)用它的成員函數(shù)func()是完全沒(méi)有問(wèn)題的,但是后面那個(gè)func2()就奇怪了。我們只知道,點(diǎn)運(yùn)算符(.)
的作用就是調(diào)用對(duì)象的成員,但是如果按照上面這個(gè)程序的字面意思來(lái)理解,就是對(duì)象object調(diào)用它的成員函數(shù)func(),然后函數(shù)func()再調(diào)用
它的成員函數(shù)func2()。這怎么能解釋得通哩??我們只知道對(duì)象有成員函數(shù),但是從來(lái)沒(méi)有聽(tīng)說(shuō)過(guò)函數(shù)也可以有成員函數(shù)的啊。沒(méi)有辦法,只有翻C++的
工具書(shū),最后,居然發(fā)現(xiàn)了這個(gè)原來(lái)就是C++中的“包含”思想。那么究竟何為包含呢,且聽(tīng)小弟慢慢敘來(lái)......^_^
何為“包含”,其實(shí)說(shuō)白了就是一個(gè)類(lèi)可以包含另一個(gè)類(lèi)的對(duì)象。即如下程序所示:
1 2 3 4 5 6 7 8 9 10 11 12 | class A
{
};
class B
{
A a;
A b;
};
|
在上面這個(gè)程序中,我們定義了類(lèi)A和類(lèi)B。其中類(lèi)B里面我們定義了類(lèi)A的兩個(gè)對(duì)象a和b。這樣的情況就叫類(lèi)B包含了類(lèi)A。下面,我們用一個(gè)程序來(lái)看一下“包含”:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 | #include <iostream>
using namespace std;
class A
{
public :
A( int i){x=i;cout<< "調(diào)用A類(lèi)的構(gòu)造函數(shù)\n" ;}
~A(){cout<< "調(diào)用A類(lèi)的析構(gòu)函數(shù)\n" ;};
void get() {cout<< "A類(lèi)中X的值為:" <<x<<endl;}
private :
int x;
};
class B
{
public :
B( int i, int j, int k):a(i),b(j),y(k){cout<< "調(diào)用B類(lèi)的構(gòu)造函數(shù)\n" ;}
A geta(){ return a;}
A getb(){ return b;}
~B(){cout<< "調(diào)用B類(lèi)的析構(gòu)函數(shù)\n" ;}
void gety(){cout<< "B類(lèi)中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;
}
|
- 首先是對(duì)兩個(gè)類(lèi)進(jìn)行分析:在上面這個(gè)程序中,我們定義了兩個(gè)類(lèi)A和B。其中可以看到,在類(lèi)B的私有成員變量里面,我們定義了一個(gè)類(lèi)B自己的成員變
量,另外還定義了兩個(gè)類(lèi)A的對(duì)象a和b(22行和23行);另外在類(lèi)B的公有函數(shù)中,我們定義了兩個(gè)返回值為類(lèi)A的函數(shù):geta()和getb(),它
們的作用就是返回在類(lèi)B中定義的兩個(gè)類(lèi)A的對(duì)象a和b。在這里我們特別應(yīng)該注意的是類(lèi)B的構(gòu)造函數(shù):
B( int i, int j, int k):a(i),b(j),y(k){cout<< "調(diào)用B類(lèi)的構(gòu)造函數(shù)\n" ;}
|
這個(gè)構(gòu)造函數(shù)很有意思。我們可以看到它不僅初始化了自己的私有成員變量y,而且也順帶初始化了類(lèi)A的兩個(gè)對(duì)象a和b。那么它肯定會(huì)調(diào)用類(lèi)A的構(gòu)造函
數(shù),而且會(huì)調(diào)用兩次。然后再調(diào)用一次B類(lèi)自己的構(gòu)造函數(shù)。那么析構(gòu)的時(shí)候順序應(yīng)該就是相反的,首先調(diào)用B類(lèi)的析構(gòu)函數(shù),然后再調(diào)用兩次A類(lèi)的析構(gòu)函數(shù)。我
們可以看到后面的程序輸出圖這樣說(shuō)滴,^_^。(見(jiàn)輸出的紅色框和黃色框)
-
現(xiàn)在我們?cè)賮?lái)看一看主函數(shù)中的東東。首先在程序的第29行,我們定義了B類(lèi)的對(duì)象b,并調(diào)用了A類(lèi)和B類(lèi)的構(gòu)造函數(shù)初始化了類(lèi)A的對(duì)象a、b和類(lèi)B
的對(duì)象b。然后在程序的第30行我們就可以看到在博文一開(kāi)始介紹的Qt中的東東。這里我們就搞不懂了,它們到底是干啥用的,什么都不說(shuō)了,先看一下運(yùn)行結(jié)
果:

我們可以看到,返回的類(lèi)A的對(duì)象a中x的值為1,另外一個(gè)類(lèi)A的對(duì)象b中x的值為2。好了,豁然開(kāi)朗了,我們來(lái)解釋一下程序第30行和31行。第29行用
類(lèi)B的對(duì)象b來(lái)調(diào)用成員函數(shù)geta(),該函數(shù)是在17行定義的,作用就是返回類(lèi)A的對(duì)象a,因此第30行
就相當(dāng)于
這里的原因就是,因?yàn)閎.geta()返回的是x的成員值為1的對(duì)象a,所以再調(diào)用類(lèi)A的get()函數(shù)就可以省略了對(duì)象a了。那么同理,程序第31行就相當(dāng)于
這就說(shuō)明了類(lèi)B可以通過(guò)成員函數(shù)來(lái)訪(fǎng)問(wèn)被包含的類(lèi)A對(duì)象的成員變量。就是文章一開(kāi)始提到的方式,其實(shí)它是隱藏了聲明類(lèi)A的對(duì)象,因?yàn)橛捎诎脑?,?lèi)B已經(jīng)幫類(lèi)A搞定了對(duì)象的聲明了~~~
OK咯,解決了Qt這個(gè)問(wèn)題,我也順帶地搞懂了包含這個(gè)概念,嘿嘿......好了,收工?。?!
|