萬盛學電腦網

 萬盛學電腦網 >> 網絡編程 >> 編程語言綜合 >> 函數指針與指針函數的學習總結

函數指針與指針函數的學習總結

函數指針是指向函數的指針,指針函數是指一個函數的返回值是一個指針,但下面的幾道題還是感覺很迷惑。各位能否講的詳細點呢?

(1) float(**def)[10]   def是什麼?
(2) double*(*gh)[10]   gh是什麼?
(3) double(*f[10])()   f是什麼?
(4) int*((*b)[10])    b是什麼?這樣老感覺有點亂,有什麼竅門可以記得並理解的清楚一點麼?

======================
解答:
  
(1) def是一個指針, 指向的對象也是一個指針, 指向的指針最終指向的是10個float構成的數組.

(2) gh是指針, 指向的是10個元素構成的數組, 數組的元素是double*類型的指針.

(3) f是10個元素構成的數組, 每個元素是指針, 指針指向的是函數, 函數類型為無參數且返回值為double. 下面要講的竅門的例子跟這個很類似.

(4) b是指針,指向的是10個元素構成的數組, 數組元素為int*類型的指針.

竅門如下:
如果我們碰到復雜的類型聲明,該如何解析它?例如:
char (*a[3])(int);
a到底被聲明為什麼東東?指針?數組?還是函數?

分析時,從a 最接近(按運算符優先級)處開始。我們看到a最接近符號是[ ]——注意:*比[ ]的優先級低。a後既然有[ ],那麼a是數組,而且是包含3個元素的數組。

那 這個數組的每個元素是什麼類型呢?雖然數組a只含有a[0]、a[1]、a[2]三個元素,a[3]實際上已經越界,但在分析數組a的元素的類型時,我們 正好需要形式上的元素a[3]。知道了a[3]的類型,就知道了a的元素的類型。 a[3]是什麼類型?是指針,因為它的前面有*. 由此可知,數組a的元素是指針。

光說是指針還不夠。對於指針,必須說出它指向的東東是什麼類型。它指向的東東是什麼,就看*a[3]是什 麼(a[3]是指針,它指向的東東當然是*a[3])了。繼續按優先級觀察,我們看到*a[3]後面有小括號,所以可以肯定*a[3]是函數。即數組a的 元素是指向函數的指針。

指向的是什麼類型的函數?這很明顯,是入參為int、返回值為char的類型的函數。
至此解析完畢。
按上述方法,再復雜的也可以一步步解析出來。

就像習武不是為了打人而是為了防身一樣,我們了解上述方法是為了看懂別人寫的復雜聲明,而不是為了在實踐中自己去構造這種復雜的東東。實在需要復雜聲明時,可以用typedef替代一部分。例如上面語句可改成兩句:
typedef char (*FUN_PTR)(int);
FUN_PTR a[3];
這樣就清晰多了。
此外,上面的分析方法還讓我們對某些東西的本質更加清楚。比如,n維數組的本質都是一維數組。看個具體的例子:
int a[3][5];
這句聲明的是一個包含3個元素的一維數組,其每個元素又是一個由5個int數構成的數組。我們不能理解為:a是一個包含5個元素的一維數組,其每個元素又是一個由3個int數構成的數組。為什麼?還是按上面的方法分析,這裡從略。

有的書上或網上提供"向右看,向左看"的方法, 其實缺乏通用性, 比如它不適用於對多維數組本質的分析. 而且這種方法掩蓋了本質. 本質應該是按上面所講的,根據運算符優先級逐層剝開.

  ==============================================================================
 
一、指針函數
當一個函數聲明其返回值為一個指針時,實際上就是返回一個地址給調用函數,以用於需要指針或地址的表達式中。
格式:
類型說明符 * 函數名(參數)
當然了,由於返回的是一個地址,所以類型說明符一般都是int。
例如:int *GetDate();
      int * aaa(int,int);
函數返回的是一個地址值,經常使用在返回數組的某一元素地址上。

復制代碼 代碼如下:
        int * GetDate(int wk,int dy);
        main()
        {
            int wk,dy;
            do
            {
                printf(Enter week(1-5)day(1-7)/n);
                scanf(%d%d,&wk,&dy);
            }
            while(wk<1||wk>5||dy<1||dy>7);
            printf(%d/n,*GetDate(wk,dy));
        }

        int * GetDate(int wk,int dy)
        {
            static int calendar[5][7]=
            {
               {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,-1}
            };
            return &calendar[wk-1][dy-1];
        }

       
程序應該是很好理解的,子函數返回的是數組某元素的地址。輸出的是這個地址裡的值。

二、函數指針
指向函數的指針包含了函數的地址,可以通過它來調用函數。聲明格式如下:
類型說明符 (*函數名)(參數)
其實這裡不能稱為函數名,應該叫做指針的變量名。這個特殊的指針指向一個返回整型值的函數。指針的聲明必須和它指向函數的聲明保持一致。
指針名和指針運算符外面的括號改變了默認的運算符優先級。如果沒有圓括號,就變成了一個返回整型指針的函數的原型聲明。
例如:
    void (*fptr)();
把函數的地址賦值給函數指針,可以采用下面兩種形式:
        fptr=&Function;
        fptr=Function;
取地址運算符&不是必需的,因為單單一個函數標識符就標號表示了它的地址,如果是函數調用,還必須包含一個圓括號括起來的參數表。
可以采用如下兩種方式來通過指針調用函數:
        x=(*fptr)();
        x=fptr();
第二種格式看上去和函數調用無異。但是有些程序員傾向於使用第一種格式,因為它明確指出是通過指針而非函數名來調用函數的。下面舉一個例子:

復制代碼 代碼如下:
        void (*funcp)();
        void FileFunc(),EditFunc();
        main()
        {
            funcp=FileFunc;
            (*funcp)();
            funcp=EditFunc;
            (*funcp)();
        }

        void FileFunc()
        {
            printf("FileFunc/n");
        }

        void EditFunc()
      

copyright © 萬盛學電腦網 all rights reserved