1. 程式人生 > >156_深入理解指標—>指標函式與函式指標的區別

156_深入理解指標—>指標函式與函式指標的區別

深入理解指標—>指標函式與函式指標的區別

2014年11月16日 20:13:46 _Tham 閱讀數:884 標籤: C語言指標指標陣列指標函式函式指標指標的指標更多

個人分類: C/C++

版權宣告:本文為博主原創文章,未經博主允許不得轉載。 https://blog.csdn.net/txl16211/article/details/41175743

一、

在學習過程中發現這“指標函式”與“函式指標”容易搞錯,所以今天,我自己想一次把它搞清楚,找了一些資料,首先它們之間的定義:

1、指標函式是指帶指標的函式,即本質是一個函式。函式返回型別是某一型別的指標

   型別識別符號    *函式名(引數表)

   int *f(x,y);

首先它是一個函式,只不過這個函式的返回值是一個地址值。函式返回值必須用同類型的指標變數來接受,也就是說,指標函式一定有函式返回值,而且,在主調函式中,函式返回值必須賦給同類型的指標變數。

表示:

float *fun();

float *p;

p = fun(a);

注意指標函式與函式指標表示方法的不同,千萬不要混淆。最簡單的辨別方式就是看函式名前面的指標*號有沒有被括號()包含,如果被包含就是函式指標,反之則是指標函式。

來講詳細一些吧!請看下面

 

指標函式:

    當一個函式宣告其返回值為一個指標時,實際上就是返回一個地址給呼叫函式,以用於需要指標或地址的表示式中。
    格式:
          型別說明符  * 函式名(引數)
    當然了,由於返回的是一個地址,所以型別說明符一般都是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));

 

           //GetDate(int ,int )函式返回的是一個地址(指標),因此使用*取地址裡的值)
        }

        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];
        }
    程式應該是很好理解的,子函式返回的是陣列某元素的地址。輸出的是這個地址裡的值

 

 函式指標

2、函式指標是指向函式的指標變數,即本質是一個指標變數

   int (*f) (int x); /* 宣告一個函式指標 */

   f=func; /* 將func函式的首地址賦給指標f */

 

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

        void (*funcp)();
        void FileFunc(),EditFunc();

        main()
        {
            funcp=FileFunc;  //將函式FileFunc()地址賦給funcp
            (*funcp)();           //通過函式指標(*funcp)()執行所指向的函式FileFunc(),執行操作
            funcp=EditFunc;
            (*funcp)();
        }

        void FileFunc()
        {
            printf(FileFunc\n);
        }

        void EditFunc()
        {
            printf(EditFunc\n);
        }

        程式輸出為:
            FileFunc
            EditFunc

 

主要的區別是一個是指標變數,一個是函式。在使用是必要要搞清楚才能正確使用

 

 

二、指標的指標

       指標的指標看上去有些令人費解。它們的宣告有兩個星號。例如:
       char  ** cp;
       如果有三個星號,那就是指標的指標的指標,四個星號就是指標的指標的指標的指標,依次類推。當你熟悉了簡單的例子以後,就可以應付複雜的情況了。當然,實際程式中,一般也只用到  二級指標,三個星號不常見,更別說四個星號了。
       指標的指標需要用到指標的地址。
        char c='A';
        char *p=&c;
        char **cp=&p;
        通過指標的指標,不僅可以訪問它指向的指標,還可以訪問它指向的指標所指向的資料。下面就是幾個這樣的例子:
        char *p1=*cp;
        char c1=**cp;
        你可能想知道這樣的結構有什麼用。利用指標的指標可以允許被呼叫函式修改區域性指標變數和處理指標陣列。

        void FindCredit(int **);

        main()
        {
            int vals[]={7,6,5,-4,3,2,1,0};
            int *fp=vals;
            FindCredit(&fp);
            printf(%d\n,*fp);
        }

        void FindCredit(int ** fpp) //因為fpp引數傳過來是地址的地址(二重指標),所以用**呼叫資料
        {
            while(**fpp!=0)
            if(**fpp<0) break;
            else (*fpp)++;                //*fpp 訪問表示陣列的地址
        }

       首先用一個數組的地址初始化指標fp,然後把該指標的地址作為實參傳遞給函式FindCredit()。FindCredit()函式通過表示式**fpp間接地得到陣列中的資料。為遍歷陣列以找到一個負值,FindCredit()函式進行自增運算的物件是呼叫者的指向陣列的指標,而不是它自己的指向呼叫者指標的指標。語句(*fpp)++就是對形參指標指向的指標進行自增運算的(相當於應用陣列的地址上)。但是因為*運算子高於++運算子,所以圓括號在這裡是必須的,如果沒有圓括號,那麼++運算子將作用於二重指標fpp上。

三、指向指標陣列的指標

       指標的指標另一用法舊處理指標陣列。有些程式設計師喜歡用指標陣列來代替多維陣列,一個常見的用法就是處理字串。

        char *Names[]=
        {
             Bill,
             Sam,
             Jim,
             Paul,
             Charles,
             0
        };

        main()
        {
            char **nm=Names;
            while(*nm!=0) printf(%s\n,*nm++);
        }

       先用字元型指標陣列Names的地址來初始化指標nm。每次printf()的呼叫都首先傳遞指標nm指向的字元型指標,然後對nm進行自增運算使其指向陣列的下一個元素(還是指標)。注意完成上述認為的語法為*nm++,它首先取得指標指向的內容,然後使指標自增。
       注意陣列中的最後一個元素被初始化為0,while迴圈以次來判斷是否到了陣列末尾。具有零值的指標常常被用做迴圈陣列的終止符。程式設計師稱零值指標為空指標(NULL)。採用空指標作為終止符,在樹種增刪元素時,就不必改動遍歷陣列的程式碼,因為此時陣列仍然以空指標作為結束。