posts - 134,comments - 22,trackbacks - 0

          C 語言所有復(fù)雜的指針聲明,都是由各種聲明嵌套構(gòu)成的。如何解讀復(fù)雜指針聲明呢?右左法則是一個(gè)既著名又常用的方法。不過,右左法則其實(shí)并不是C 標(biāo)準(zhǔn)里面的內(nèi)容,它是從C 標(biāo)準(zhǔn)的聲明規(guī)定中歸納出來的方法。C 標(biāo)準(zhǔn)的聲明規(guī)則,是用來解決如何創(chuàng)建聲明的,而右左法則是用來解決如何辯識(shí)一個(gè)聲明的,兩者可以說是相反的。右左法則的英文原文是這樣說的:
           
                  The   right-left   rule:   Start   reading   the   declaration   from   the   innermost   parentheses,   go   right,   and   then   go   left.   When   you   encounter  parentheses,   the   direction   should   be   reversed.   Once   everything   in   the   parentheses   has   been   parsed,   jump   out  of   it.   Continue   till   the   whole   declaration   has  been   parsed.  
           
           
          這段英文的翻譯如下:
           
                   右左法則:首先從最里面的圓括號(hào)看起,然后往右看,再往左看。每當(dāng)遇到圓括號(hào)時(shí),就應(yīng)該掉轉(zhuǎn)閱讀方向。一旦解析完圓括號(hào)里面所有的東西,就跳出圓括號(hào)。重復(fù)這個(gè)過程直到整個(gè)聲明解析完畢。
           
          筆者要對(duì)這個(gè)法則進(jìn)行一個(gè)小小的修正,應(yīng)該是從未定義的標(biāo)識(shí)符開始閱讀,而不是從括號(hào)讀起,之所以是未定義的標(biāo)識(shí)符,是因?yàn)橐粋€(gè)聲明里面可能有多個(gè)標(biāo)識(shí)符,但未定義的標(biāo)識(shí)符只會(huì)有一個(gè)。
           
            現(xiàn)在通過一些例子來討論右左法則的應(yīng)用,先從最簡(jiǎn)單的開始,逐步加深:
           
          int   (*func)(int   *p);
           
          首先找到那個(gè)未定義的標(biāo)識(shí)符,就是func ,它的外面有一對(duì)圓括號(hào),而且左邊是一個(gè)* 號(hào),這說明func 是一個(gè)指針,然后跳出這個(gè)圓括號(hào),先看右邊,也是一個(gè)圓括號(hào),這說明(*func) 是一個(gè)函數(shù),而func 是一個(gè)指向這類函數(shù)的指針,就是一個(gè)函數(shù)指針,這類函數(shù)具有int* 類型的形參,返回值類型是int 。
           
          int   (*func)(int   *p,   int   (*f)(int*));
           
          func 被一對(duì)括號(hào)包含,且左邊有一個(gè)* 號(hào),說明func 是一個(gè)指針,跳出括號(hào),右邊也有個(gè)括號(hào),那么func 是一個(gè)指向函數(shù)的指針,這類函數(shù)具有int   * 和int   (*)(int*) 這樣的形參,返回值為int 類型。再來看一看func 的形參int   (*f)(int*) ,類似前面的解釋,f 也是一個(gè)函數(shù)指針,指向的函數(shù)具有int* 類型的形參,返回值為int 。
           
          int   (*func[5])(int   *p);
           
          func 右邊是一個(gè)[] 運(yùn)算符,說明func 是一個(gè)具有5 個(gè)元素的數(shù)組,func 的左邊有一個(gè)* ,說明func 的元素是指針,要注意這里的* 不是修飾func 的,而是修飾func[5] 的,原因是[] 運(yùn)算符優(yōu)先級(jí)比* 高,func 先跟[] 結(jié)合,因此* 修飾的是func[5] 。跳出這個(gè)括號(hào),看右邊,也是一對(duì)圓括號(hào),說明func 數(shù)組的元素是函數(shù)類型的指針,它所指向的函數(shù)具有int* 類型的形參,返回值類型為int 。
            int   (*(*func)[5])(int   *p);
           
          func 被一個(gè)圓括號(hào)包含,左邊又有一個(gè)* ,那么func 是一個(gè)指針,跳出括號(hào),右邊是一個(gè)[] 運(yùn)算符號(hào),說明func 是一個(gè)指向數(shù)組的指針,現(xiàn)在往左看,左邊有一個(gè)* 號(hào),說明這個(gè)數(shù)組的元素是指針,再跳出括號(hào),右邊又有一個(gè)括號(hào),說明這個(gè)數(shù)組的元素是指向函數(shù)的指針。總結(jié)一下,就是:func 是一個(gè)指向數(shù)組的指針,這個(gè)數(shù)組的元素是函數(shù)指針,這些指針指向具有int* 形參,返回值為int 類型的函數(shù)。
           
          int   (*(*func)(int   *p))[5];
           
          func 是一個(gè)函數(shù)指針,這類函數(shù)具有int* 類型的形參,返回值是指向數(shù)組的指針,所指向的數(shù)組的元素是具有5 個(gè)int 元素的數(shù)組。
           
          要注意有些復(fù)雜指針聲明是非法的,例如:
           
          int   func(void)   [5];
           
          func 是一個(gè)返回值為具有5 個(gè)int 元素的數(shù)組的函數(shù)。但C 語言的函數(shù)返回值不能為數(shù)組,這是因?yàn)槿绻试S函數(shù)返回值為數(shù)組,那么接收這個(gè)數(shù)組的內(nèi)容的東西,也必須是一個(gè)數(shù)組,但C 語言的數(shù)組名是一個(gè)右值,它不能作為左值來接收另一個(gè)數(shù)組,因此函數(shù)返回值不能為數(shù)組。
           
          int   func[5](void);
           
          func 是一個(gè)具有5 個(gè)元素的數(shù)組,這個(gè)數(shù)組的元素都是函數(shù)。這也是非法的,因?yàn)閿?shù)組的元素除了類型必須一樣外,每個(gè)元素所占用的內(nèi)存空間也必須相同,顯然函數(shù)是無法達(dá)到這個(gè)要求的,即使函數(shù)的類型一樣,但函數(shù)所占用的空間通常是不相同的。
           
          作為練習(xí),下面列幾個(gè)復(fù)雜指針聲明給讀者自己來解析,答案放在第十章里。
           
          int   (*(*func)[5][6])[7][8];
           
          int   (*(*(*func)(int   *))[5])(int   *);
           
          int   (*(*func[7][8][9])(int*))[5];
           
                   實(shí)際當(dāng)中,需要聲明一個(gè)復(fù)雜指針時(shí),如果把整個(gè)聲明寫成上面所示的形式,對(duì)程序可讀性是一大損害。應(yīng)該用typedef 來對(duì)聲明逐層分解,增強(qiáng)可讀性,例如對(duì)于聲明:
           
          int   (*(*func)(int   *p))[5];
           
          可以這樣分解:
           
          typedef     int   (*PARA)[5];
          typedef   PARA   (*func)(int   *);
           
          這樣就容易看得多了。 


          本文來自CSDN博客,轉(zhuǎn)載請(qǐng)標(biāo)明出處:http://blog.csdn.net/xiaojianpitt/archive/2009/10/30/4748427.aspx

          posted on 2010-05-04 21:33 何克勤 閱讀(182) 評(píng)論(0)  編輯  收藏 所屬分類: C/C++
          主站蜘蛛池模板: 神池县| 景德镇市| 达拉特旗| 巴彦县| 海林市| 延长县| 株洲县| 河西区| 长子县| 固始县| 岳阳县| 安乡县| 平凉市| 清水河县| 大田县| 新巴尔虎左旗| 鲁甸县| 威信县| 汉寿县| 武平县| 公主岭市| 承德市| 潞西市| 潼南县| 英山县| 南皮县| 绥宁县| 峨边| 栖霞市| 合水县| 定远县| 贵州省| 太谷县| 武城县| 长治市| 石台县| 杭州市| 厦门市| 莲花县| 永康市| 永兴县|