• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            f(sixleaves) = sixleaves

            重劍無(wú)鋒 大巧不工

              C++博客 :: 首頁(yè) :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
              95 隨筆 :: 0 文章 :: 7 評(píng)論 :: 0 Trackbacks
            函數(shù)指針是一個(gè)重難點(diǎn),看完書本后,決定寫篇自己做下總結(jié)。
            首先在C++\C中,函數(shù)的函數(shù)名本身就是地址,而函數(shù)指針就是存儲(chǔ)這個(gè)地址的變量。
            如下代碼void fun(int a, int b) {.....};函數(shù),其fun就是一個(gè)指針,也就是存的是地址,
            而void (*p)(int, int) = fun;就是指向這個(gè)函數(shù)的指針,其實(shí)說(shuō)來(lái)p函數(shù)指針這時(shí)也使指
            向這個(gè)函數(shù)的,所以要使用這個(gè)函數(shù)的話,理論上應(yīng)該寫成(*p)(2, 3);這種形式,但是實(shí)
            際上,通過(guò)p(2, 3)也可以成功調(diào)用,所以我們可以把函數(shù)指針在某種情況下當(dāng)成函數(shù)的別
            名,雖然這樣不符合邏輯,當(dāng)然,如果你比較較真,可以使用(*p)(2, 3);這邊講了這么多
            現(xiàn)在來(lái)總結(jié)下如何聲明一個(gè)特定類型的函數(shù)指針吧。
            知識(shí)點(diǎn)1:
            聲明特定類型的函數(shù)指針:
                 1.先寫出要聲明的指定函數(shù)的原型prototype。如void fun(int , int );
                 2.把prototype中的函數(shù)名替換成(*p),如void (*p)(int , int );
            經(jīng)過(guò)上面兩步,你就聲明了一個(gè)指向無(wú)返回值,帶有兩個(gè)int類型的函數(shù)的函數(shù)指針p。
            哇,原來(lái)函數(shù)指針這么簡(jiǎn)單啊、誰(shuí)說(shuō)指針很復(fù)雜的,誰(shuí)說(shuō)指針很難的,拉出去斬了、欺騙我們幼小的心靈。
            此時(shí)你可能會(huì)問(wèn),怎么調(diào)用啊,怎么調(diào)用啊,首先你要給他賦值,其次,調(diào)用分兩種,就是上面一開始說(shuō)
            的那兩種。說(shuō)白了,也就是對(duì)于函數(shù)指針調(diào)用函數(shù),你可以解引用,可以刻直接用地址。

            知識(shí)點(diǎn)2:
            聲明特定類型的函數(shù)指針數(shù)組:
                1.和知識(shí)點(diǎn)一1,2步一樣,我們先寫出一個(gè)。第二步在想辦法寫成一個(gè)數(shù)組
                2.把(*p)替換成(*p[3])這樣p就是一個(gè)包含3個(gè)函數(shù)指針的數(shù)組。也就是說(shuō)p是函數(shù)指針的指針!
            什么意思!什么叫做指針的指針,你在說(shuō)什么!,哈哈有得人看到這開始暈了,那么后面的你更暈,其實(shí)說(shuō)白了p是指向數(shù)組的第一個(gè)元素
            也就是說(shuō)p的地址值是第一個(gè)元素的地址,所以說(shuō)p是函數(shù)指針的指針啊,因?yàn)閜指向的時(shí)函數(shù)指針,函數(shù)指針指向的才是內(nèi)存中函數(shù)指令區(qū)域的那個(gè)塊!

            知識(shí)點(diǎn)3:
            對(duì)于數(shù)組p[n]區(qū)別,p和&p的重大區(qū)別:
            前言:我們知道p指向數(shù)組的第一個(gè)元素,所以p等價(jià)于&p[0],所以我們可以輕易看出區(qū)別了,p和&p
                    的相同點(diǎn)是在數(shù)字上,他們相同,但是在大小上,或者說(shuō)類型上,他們不同,&p指向的時(shí)整個(gè)數(shù)
                    組,如果&p + 1則跨越的時(shí)整個(gè)數(shù)組。其實(shí)我們可以從指針定義的運(yùn)算來(lái)理解,我們知道,指針
                    的加加,本質(zhì)上是地址的跨越,而跨越的長(zhǎng)度,取決于地址的類型,&p是指向數(shù)組的指針,所以
                    其跨越的長(zhǎng)度,肯定是一整個(gè)數(shù)組,而p指向的時(shí)數(shù)組中得第一個(gè)元素,所以p + 1,跨越的時(shí)數(shù)
                    組的一個(gè)元素。
            1.p和&p的相同點(diǎn),在于數(shù)值上,他們都是那個(gè)內(nèi)存塊的地址,而那個(gè)內(nèi)存塊用一個(gè)地址標(biāo)志,所以他們的數(shù)字相同。
            2.p和&p的區(qū)別在于類型上,p是指向一個(gè)元素的,&p是指向一個(gè)數(shù)組的,所以p + 1和&p + 1有十分大得區(qū)別。

            知識(shí)點(diǎn)4:
            問(wèn)題:
            假設(shè)有const double * (*pa[3])(const double * , int ) = {f1, f2, f3},聲明指向該函數(shù)指針數(shù)組的指針。

            有了知識(shí)點(diǎn)3,理解知識(shí)點(diǎn)4就十分容易了。
                  1.C11方法auto pc = &pa;  C11的方式十分簡(jiǎn)單方便但是對(duì)于不支持C11的請(qǐng)看第二個(gè),最原始的方法
                  2.分析:因?yàn)槲覀円暶鞯臅r(shí)一個(gè)指針,而不是一個(gè)數(shù)組所以首先用(*pd)把其擴(kuò)起來(lái),然后其是指向一個(gè)函數(shù)指針數(shù)組,這個(gè)數(shù)組有3個(gè)元素。所以其核心部分就是(*pd)[3],此時(shí)的意思就是所pd是一個(gè)指針,其指向包含3個(gè)元素的數(shù)組
            所以第三步我們要說(shuō)明,數(shù)組元素的類型了,類型就是const double * (*)(const double *, int),也就是const double * (* (*pd)[3])(const double *, int)。
            tips:
            有人反映第三步類型看不懂,其實(shí)很簡(jiǎn)單,我舉個(gè)例子,
            1.int *p;p的類型是int *,
            2.而int *p[3],p的類型是int * [3]也很好理解就是帶有三個(gè)數(shù)組,元素類型都是int *。
            3.而int (*p[3])(int ),p的類型是int (*[3])(int),p是指向一個(gè)3個(gè)元素的數(shù)組,且每個(gè)元素的類型都是int (*)(int)。
            (指針是什么類型主要看括號(hào)、*號(hào)、[]號(hào),和參數(shù)列表的結(jié)合順序,如果只有(*p)(int ,....)那p就是函數(shù)指針)
            所以有以下推論。
                  
            元素的數(shù)據(jù)類型推論:
            要判斷數(shù)組或指針的類型,就是拿掉<數(shù)組名、指針名>后組成的,而且[]和*是從右向左結(jié)合。
            要判斷數(shù)組元素的類型,就是在數(shù)組類型的基礎(chǔ)上把大小拿掉


            知識(shí)點(diǎn)5:
            使用typedef創(chuàng)建函數(shù)指針類型別名:
                   1.typedef簡(jiǎn)化函數(shù)指針其本質(zhì)就是為函數(shù)指針的類型取別名。
                   如下:
                   typedef const double *(*p_fun)(const double *, int);其是此時(shí)相當(dāng)于typedef Const double *(*)(const double *, int)  p_fun;
            當(dāng)然你不能這樣寫,這樣寫只是方便你理解。
                   p_fun不是函數(shù)指針了,而是這種函數(shù)指針類型的別名,所以此時(shí)你可以這樣做
                   p_fun p1 = f1;
                   p_fun pa[3] = {f1, f2, f3};
                   p_fun (*pd)[3] = &pa;
                   是不是比之前的簡(jiǎn)單許多!,這個(gè)很重要哦
            好了函數(shù)指針就總結(jié)到這,如有不足請(qǐng)指教。
            以下是測(cè)試代碼:測(cè)試代碼參考自C++ Primer Plus
             1 //
             2 //
             3 //  Unit7
             4 //
             5 //  Created by sixleaves on 14-7-26.
             6 //  Copyright (c) 2014年 sixleaves. All rights reserved.
             7 //
             8 
             9 #include <iostream>
            10 
            11 const double * f1(const double ar[], int n);
            12 const double * f2(const double [], int);
            13 const double * f3(const double *, int);
            14 
            15 int main() {
            16     
            17     using namespace std;
            18     double av[3] = {1112.3, 1542.6, 2227.9};
            19     
            20     
            21     //pointer to a function
            22     const double * (*p1)(const double *, int) = f1;
            23     auto p2 = f2;//C++11 automatic type deduction
            24     //pre-C11 can use the following code instead
            25     //const double * (*p2)(const double *, int) = f2;
            26     
            27     
            28     cout << "Using pointers to functions:\n";
            29     cout << "Address Value\n";
            30     cout << (*p1)(av, 3) << ": " << *(*p1)(av, 3) << endl;//嚴(yán)密的邏輯性調(diào)用
            31     cout << p2(av, 3) << ": " << *p2(av, 3) << endl;//感性調(diào)用
            32     
            33     //pa an array of pointers
            34     //auto doesn't work with list initialization
            35     const double *(*pa[3])(const double *, int) = {f1, f2, f3};
            36     //but it does work for initializing to a single value
            37     //pb a pointer to  fitst element of pa
            38     auto pb = pa;
            39     //pre-c11 can use the following code instead
            40     //const double *(**pb)(const double *, int) = pa;
            41     
            42     cout << "\nUsing an array of pointer to functions:\n";
            43     cout << " Address Value\n";
            44     for (int i = 0; i < 3; i++) {
            45         cout << pb[i](av, 3) << ": " << *pb[i](av, 3) <<endl;
            46     }
            47     
            48     //what about a pointer to an array of function pointers;
            49     cout << "\nUsing pointers to an array of function pointers:\n";
            50     cout << " Adress Values\n";
            51     //easy way to declare pc
            52     auto pc = &pa;
            53     //pre c11 can use the following code instead
            54     //const double *(*(*pc)[3])(const double *, int) = &pa;
            55     
            56     cout << (*pc)[0](av, 3) << ": " << *(*pc)[0](av, 3) << endl;
            57     
            58     //hard way to declare pd;
            59     const double *(*(*pd)[3])(const double *, int) = &pa;
            60     
            61     //store return value in pdb;
            62     const double *pdb = (*pd)[1](av, 3);
            63     cout << pdb << ": " << *pdb << endl;
            64     
            65     //alternative notation
            66     cout << (*(*pd)[2])(av, 3) << ": " << *(*(*pd)[2])(av, 3) << endl;
            67     
            68     return 0;
            69 }
            70 const double * f1(const double ar[], int n) {
            71     return ar;
            72 }
            73 const double * f2(const double ar[], int) {
            74     return ar + 1;
            75 }
            76 const double * f3(const double ar[], int) {
            77     return ar + 2;
            78 }
            79 
            posted on 2014-07-26 18:08 swp 閱讀(2474) 評(píng)論(5)  編輯 收藏 引用 所屬分類: program language

            評(píng)論

            # re: 深入探討函數(shù)指針[未登錄](méi) 2014-08-13 22:12 seahouse
            寫的真好。
            知識(shí)點(diǎn)4 沒(méi)怎么看明白。。  回復(fù)  更多評(píng)論
              

            # re: 深入探討函數(shù)指針 2014-08-17 11:22 weiki
            主要是看不懂聲明的那一串代碼是怎么回事,只要涉及多級(jí)指針,感覺很難理解的樣子  回復(fù)  更多評(píng)論
              

            # re: 深入探討函數(shù)指針 2014-08-18 12:34 swp
            哪一串?@weiki
              回復(fù)  更多評(píng)論
              

            # re: 深入探討函數(shù)指針 2014-08-18 12:34 swp
            我修改下,你在重新看一篇。@seahouse
              回復(fù)  更多評(píng)論
              

            # re: 深入探討函數(shù)指針 2015-01-10 18:36 skndr
            寫的好  回復(fù)  更多評(píng)論
              

            国产成人AV综合久久| 亚洲国产精品一区二区三区久久| 久久综合综合久久综合| 人妻精品久久久久中文字幕69| 色成年激情久久综合| 久久综合视频网站| 久久99精品久久久久久动态图| 一本久久综合亚洲鲁鲁五月天| 亚洲AV日韩精品久久久久久久| 亚洲国产精品久久久久婷婷软件| 噜噜噜色噜噜噜久久| 狠狠色噜噜狠狠狠狠狠色综合久久| 久久久久久久尹人综合网亚洲 | 综合久久国产九一剧情麻豆| 久久精品亚洲一区二区三区浴池| 大美女久久久久久j久久| 少妇久久久久久久久久| 日本加勒比久久精品| 国产精品一久久香蕉国产线看| 亚洲欧美国产日韩综合久久 | 日韩美女18网站久久精品| 久久99国产精品久久99果冻传媒| 久久se精品一区精品二区国产| 久久国产色AV免费观看| 亚洲AV日韩精品久久久久久| 久久妇女高潮几次MBA| 久久精品免费大片国产大片| 久久久九九有精品国产| 99精品伊人久久久大香线蕉| 久久久免费精品re6| 国内精品久久久久久久久电影网| 少妇人妻综合久久中文字幕| 九九久久精品无码专区| 品成人欧美大片久久国产欧美... 品成人欧美大片久久国产欧美 | 国内精品久久久久影院优| 久久狠狠高潮亚洲精品| 久久亚洲精品成人AV| 久久国产亚洲精品无码| 久久99国产精品尤物| 欧美久久精品一级c片片| 久久国产精品99精品国产987|