• <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>

              C++博客 :: 首頁 :: 聯(lián)系 ::  :: 管理
              163 Posts :: 4 Stories :: 350 Comments :: 0 Trackbacks

            常用鏈接

            留言簿(48)

            我參與的團隊

            搜索

            •  

            積分與排名

            • 積分 - 399102
            • 排名 - 59

            最新評論

            閱讀排行榜

            評論排行榜

            一、 數(shù)組的指針、指針數(shù)組以及指向指針的指針

              考慮數(shù)組的指針的時候我們要同時考慮類型和維數(shù)這兩個屬性。換一句話,就是說一個數(shù)組排除在其中存儲的數(shù)值,那么可以用類型和維數(shù)來位置表示他的種類。

            A)一維數(shù)組
              在c和c++中數(shù)組的指針就是數(shù)組的起始地址(也就第一個元素的地址),而且標準文檔規(guī)定數(shù)組名代表數(shù)組的地址(這是地址數(shù)值層面的數(shù)組表示)。例如:

            1int a[10];
            2int *p;

            p=&a[0]//和p=a是等價的:

            因為a是數(shù)組名,所以他是該數(shù)組的地址,同時因為第一個元素為a[0],那么&a[0]也代表了該數(shù)組的地址。但是我們是不是就說一個數(shù)組名和 該數(shù)組的第一個元素的&運算是一回事呢?在一維的時候當時是的,但是在高維的時候,我們要考慮到維數(shù)給數(shù)組帶來的影響。
              a[10]是 一個數(shù)組,a是數(shù)組名,它是一個包含10個int類型的數(shù)組類型,不是一般的指針變量噢!(雖然標準文檔規(guī)定在c++中從int[]到int*直接轉換是 可以的,在使用的時候似乎在函數(shù)的參數(shù)為指針的時候,我們將該數(shù)組名賦值沒有任何異樣),a代表數(shù)組的首地址,在數(shù)字層面和a[10]的地址一樣。這樣我 們就可以使用指針變量以及a來操作這個數(shù)組了。
            所以我們要注意以下問題:

            (1) p[i]和a[i]都是代表該數(shù)組的第i+1個元素;
            (2) p+i和a+i代表了第i+1個元素的地址,所以我們也可以使用 *(p+I)和*(a+I)來引用對象元素;
            (3)p+1不是對于指針數(shù)量上加一,而是表示從當前的位置跳過當前指針指向類型長度的空間,對于win32的int為4byte;

            B)多維數(shù)組
            對于二維數(shù)組a[4][6];由于數(shù)組名代表數(shù)組的起始地址,所以a(第一層)和第一個元素a[0][0]地址的數(shù)字是相同的,但是意義卻是不同的。對 于該數(shù)組我們可以理解為:a的一維數(shù)組(第一層),它有四個元素a[0]、a[1]、a[2]、a[3](第二層),而每個元素又含有6個元素a[0] [0],a[0][1],a[0][2],a[0][3],a[0][4],a[0][5](第三層),…到此我們終于訪問到了每個元素了,這個過程我們 經(jīng)歷了:a->a[0]->a[0][0];
              整體來講:a是一個4行5列的二維數(shù)組,a表示它指向的數(shù)組的首地址(第一個元素地 址&a[0]),同時a[0]指向一行,它是這個行的名字(和該行的第一個元素的首地址相同(第一個元素為地址&a[0][0]))。所 以從數(shù)字角度說:a、a[0]、&a[0][0]是相同的,但是他們所處的層次是不同的。
              既然a代表二維數(shù)組,那么a+i就表示它的第i+1個元素*(a+i)的地址,而在二維數(shù)組中
            *(a+i)又指向一個數(shù)組,*(a+i)+j表示這個數(shù)組的第j+1個元素的地址,所以要訪問這個元素可以使用 *(*(a+i)+j)(也就是a[i][j])。
            他們的示意圖為(虛線代表不是實際存在的):

            對照這個圖,如下的一些說法都是正確的(對于a[4][6]):

            • a是一個數(shù)組類型,*a指向一個數(shù)組;
            • a+i指向一個數(shù)組;
            • a、*a和&a[0][0]數(shù)值相同;
            • a[i]+j和*(a+i)+j是同一個概念;

              總結一下就是:我們對于二維指針a,他指向數(shù)組a[0,1,2,3],使用*,可以使他降級到第二層次,這樣*a就指向了第一個真正的數(shù)組。對于其他的情況我們也可以采用相同的方式,對于其他維數(shù)和類型的數(shù)組我們可以采用相類似的思想。

            說到指向數(shù)組的指針,我們還可以聲明一個指針變量讓它指向一個數(shù)組。例如:

            1int (*p)[5];

            這時p就是一個指針,要指向一個含有5個int類型元素的數(shù)組,指向其他的就會出現(xiàn)問題。
            這個時候我們可以使用上面的什么東西來初始化呢?
            我們可以使用*a,*(a+1),a[2]等。
            原因很簡單:我們在一個二維的數(shù)組中,那么表達方式有上面的相互類似的意義呢?只有 *a,*(a+1),a[2]等,

            C)指針數(shù)組
              一個指針數(shù)組是指一個數(shù)組中的每個元素都是一個指針,例如:

            int *p[10];//而不能是int (*p)[10]
            or

            char *p[10];

            此時p是一個指針(數(shù)值上和&p[0]一樣);
            在前面有int t[10];

            int * pt=t;//使用pt指向t

            那么這里我們用什么指向int *t[10]中的t呢?我們要使用一個指針的指針:

            int **pt=t;

              這是因為:在int *t[10]中,每個元素是指針,那么同時t又指向這個數(shù)組,數(shù)組上和&t[0]相同,也就是指向t[0],指向一個指針變量,可以說是一個指針的指針了,所以自然要用

            int **pt;


            D)指針的指針
            一個指針變量內(nèi)部可以存儲一個值,這個值是另外一個對象的地址,所以我們說一個指針變量可以指向一個普通變量,同樣這個指針變量也有一個地址,也就是說 有一個東西可以指向這個指針變量,然后再通過這個指針變量指向這個對象。那么如何來指向這個指針變量呢?由于指針變量本身已經(jīng)是一個指針了(右值),那么 我們這里就不能用一般的指針了,需要在指針上體現(xiàn)出來這些特點,我們需要定義指針的指針(二重指針)。

            int *p1=&i;
            int**p2=&p1;

            綜合以上的所有點,下面是我們常常看到一些匹配(也是經(jīng)常出錯的地方):

            int a[3],b[2][3],c,*d[3];
            void fun1(int *p);
            void fun2(int (*p)[3]);
            void fun3(int **p);
            void fun4(int p[3]); 
            void fun5(int p[]);
            void fun6(int p[2][3]);
            void fun7(int (&p)[3]);

            函數(shù) 不會產(chǎn)生編譯時刻的可能值(但邏輯上不一定都對)

            函數(shù)

            不會產(chǎn)生編譯時刻的可能值(但邏輯上不一定都對)

            fun1

            a, &a[i], *b ,b[i],&b[i][j] ,&c ,d[i]

            fun2

            b,b+i,

            fun3

            d

            fun4

            a, &a[i], *b ,b[i],&b[i][j] ,&c ,d[i]

            fun5

            a, &a[i], *b ,b[i],&b[i][j] ,&c ,d[i]

            fun6

              b

            fun7

            a

            為什么可以有這樣的搭配,原因如下:

            • 對 于fun1 fun4 fun 5: 在編譯器看來fun1,fun4,fun5的聲明是一樣,在編譯時候,編譯器把數(shù)組的大小舍去不考慮,只考慮它是一個指針,也就是說有沒有大小說明是一樣 的,所以三者的形式都是fun1的形式(其實只要提供了int*指針就可以了);
            • 對于fun7 :以上的解釋對于引用是不適用的,如果變量被聲明為數(shù)組的引用,那么編譯器就要考慮數(shù)組的大小了,那么必須和聲明一模一樣(所以fun7就只有a合適);
            • 對于fun2:p是一個指向一個含有3個元素的數(shù)組,這樣b和b+i正好合適,而a卻不是(它是指向a[0]的,不是指向這個數(shù)組的);
            • 對于fun3:p是一個指針的指針,而d指向d[0],同時d[0]又是一個指針,所以d就是一個指針的指針。但是b卻不是(它是一個2*3的矩陣也就是年int [2][3]類型);
            • 對于fun6,p是一個2*3的數(shù)組類型,和b恰好完全匹配;

            二、函數(shù)指針、函數(shù)的指針參數(shù)以及返回指針的函數(shù)

            A) 函數(shù)指針
            C++規(guī)定,一個函數(shù)的地址就是這個函數(shù)的名字。我們需要指出的就是一個指針需要指定類型是為了后來的指針解析時候使用,通過指針有效快速訪問對象。那 么對于函數(shù)的指針,它要表示出該函數(shù)的那些特性才能滿足解析的唯一性呢?答案就是一個函數(shù)的特性有它的參數(shù)列表和返回類型。

            下面是一個函數(shù)指針的例子:

            int (*p)(int I,int j);

            不能是

            int *p(int I,int j),

            這樣就變成了返回指針的函數(shù)聲明了。

            在C++中處于對安全性的考慮,指針和它指向的對象要類型一致,也就說上面的指針所指向的函數(shù)的特性要和它一模一樣:例如指向int min(int I,int j);是可以的。但是指向int min(double I ,double j);是不可以。函數(shù)指針也和其他的指針一樣,在使用的時候很怕發(fā)生"懸空",所以在使用的時候同樣要判斷有效性,或者在定義的時候就初始化。

            int (*p)(int I,int j)=min;
            int (*p)(int I,int j)=&min;
            int (*p)(int I,int j)=0;

            B) 函數(shù)的指針參數(shù)
              函數(shù)指針可以作函數(shù)的參數(shù):例如我們有一個積分的算法,對于不同的數(shù)學函數(shù)可以進行積分(我們這里假設函數(shù)都是一元的);
            那么我們的算法接口可以定義為:

            template<class T>
            T integrate( T lower, T upper , T (*)(T)=0 )throw(integrated_exp);
            這里的最后的參數(shù)是一個函數(shù)的指針,并且被設定缺省值為0。這個函數(shù)返回一個值,同時需要一個參數(shù)。假如加入我們有這樣的一個函數(shù):
            double line(double x){ return a*x+b;}

            那么我就可以使用了。

            函數(shù)指針還可以作為返回類型(注意不是函數(shù)!!,某個特定的函數(shù)是不可以作為返回類型的。)假設:

            typedef int (*PF)(int );
            PF getProcessMethod( );//true

            C) 返回指針的函數(shù)
            一個函數(shù)的返回是函數(shù)的重要接口之一,c++的一個重要的強大的功能就是能夠設計足夠復雜和好用的用戶自定義類型。而同時處理和傳遞這些類型也是很麻煩 的一件事情,我們不想把我們的時間都花在這些對于我們的實際工作沒有很實質幫助的拷貝上,解決這個問題就要依賴我們的接口設計:c和c++都提供了相應的 解決方案,在c++中我們可是使用引用,講他們作為函數(shù)的實際參數(shù),或者我們在函數(shù)的實際參數(shù)中使用一個指針等。同樣我們還可以使用一個函數(shù)返回一個指 針:但是這是一個很不好解決的問題!
            我們首先容易出錯的是:將一個局部變量的地址傳出來!例如:

            UserType * Process( )
            {
              UserType ut(param-list);
              //process ut;
              return &ut;//
            }

              這個變量在我們的函數(shù)結束的時候就被銷毀了,盡管地址可以傳出去,但是這個地址已經(jīng)不存在了,已經(jīng)不能使用的東西,在這個函數(shù)之外卻不知道,難免要出錯!
            同時我還會有一個比較麻煩的問題:使用new,又容易造成內(nèi)存泄露

            UserType * Process ( )
            {
              UserTpye *put=new UserType(param-list );
              //process put;
              return put;
            }

            我們在函數(shù)內(nèi)部使用了一個new,分配了一個空間,這樣傳出來也是可以!
              就是說不會發(fā)生上面的問題了。但是用戶通常都會忘記在程序的外面在把這個借來的空間還回去!內(nèi)存空間就這樣泄露了!
            可能也是這些另人無奈的問題,所以很多程序員把函數(shù)的參數(shù)設定為指針或者引用,以此來代替這種向外傳輸吧!總之,使用這種返回指針的函數(shù)要小心!

            三、類成員的指針

            類成員和一般的外部變量相互比較,不同就是它所在的域不同,這個域很重要,它決定了該變量可以使用的范圍。那么一個指針如果要指向類的成員函數(shù)或者成員 變量,那么除了要表達它的返回類型、參數(shù)列表或者類型之外,那么還要說明它所指向的變量(或者函數(shù))的域,為了說明該域我們要使用類域限定:

            class NJUPT
            {
              static double money=20000000;
              int num;
              public:
              NJUPT():num(10){};
              int get(){return num;};
              double getMoney(){reuturn money;}
            }

            我們定義成員的指針為

            int NJUPT:: *p;//指向int型成員變量
            int (NJUPt::*)p()//指向int f()型成員函數(shù)。

            為了使用這些指針,我們需要使用該類型的變量或者指針。

            NJUPT s,*ps;

            那么調用的方式為:

            cout<<s.*p;
            cout<<(s->*p)();

              這個看起來似乎很奇怪!但是只要你想到我們定義的指針被限定在了類域中了(我們在開始定義的使用使用了NJUPT:: ),這么使用也是很自然的。
              如果一個類還有一些靜態(tài)成員變量和靜態(tài)成員函數(shù),那么我是否也想這樣使用呢?
            答案是不用,靜態(tài)成員我們就可以象使用外部的普通成員一樣定義一個指針或者函數(shù)指針來訪問就可以了,究其原因主要是這個成員的類型性質決定的。

            double *p=&NJUPT::money;
            double (*p)()=&NJUPT::getMoney(): 
             

            posted on 2008-01-06 15:59 sdfasdf 閱讀(254) 評論(0)  編輯 收藏 引用
            欧美久久久久久精选9999| 久久婷婷久久一区二区三区| 无码人妻少妇久久中文字幕| 人妻精品久久久久中文字幕| 久久婷婷五月综合国产尤物app | 欧美与黑人午夜性猛交久久久| 人人狠狠综合久久亚洲高清| 亚洲女久久久噜噜噜熟女| AAA级久久久精品无码区| 久久精品青青草原伊人| 久久精品国产免费一区| 欧美伊人久久大香线蕉综合 | 狠狠色狠狠色综合久久 | 久久久久国产视频电影| 久久男人Av资源网站无码软件| 精品水蜜桃久久久久久久| 国产精品99久久免费观看| 日韩精品久久久久久久电影| 91久久九九无码成人网站 | 久久99精品九九九久久婷婷| 久久亚洲日韩精品一区二区三区| 欧美久久久久久午夜精品| 大伊人青草狠狠久久| 亚洲av日韩精品久久久久久a| 久久精品一区二区影院| 亚洲午夜精品久久久久久人妖| 奇米综合四色77777久久| 久久天天躁狠狠躁夜夜躁2014| 亚洲精品无码专区久久同性男| 久久久九九有精品国产| 精品久久久久久久久中文字幕| 久久午夜无码鲁丝片| 性做久久久久久久| 性欧美丰满熟妇XXXX性久久久| 蜜桃麻豆WWW久久囤产精品| 一级做a爰片久久毛片免费陪| 久久久久亚洲精品男人的天堂| 久久久99精品一区二区| 久久久久亚洲精品男人的天堂| 国産精品久久久久久久| 丁香久久婷婷国产午夜视频|