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

            CG@CPPBLOG

            /*=========================================*/
            隨筆 - 76, 文章 - 39, 評論 - 137, 引用 - 0
            數(shù)據(jù)加載中……

            對數(shù)組名取地址是什么?

            這兩天有人問以下有什么代碼有什么不同?

            1 int array[100];
            2 
            3 memset(array,  0, sizeof(array));
            4 memset(&array, 0, sizeof(array));

            第3行和第4行有什么不同嗎?其實從效果上來說是一樣的,但是這里要注意 array 和 &array 的類型是不同的。array 相當于 &array[0],而 &array 是一個指向 int[100] 的指針,類型是 int(*)[100]。
            以下代碼可以看出這個不同:

             1 #include <stdio.h>
             2 
             3 int main()
             4 {
             5         int array[100= {012};
             6         typedef int (*ARRAY)[100];
             7         int* p1 = array;
             8         ARRAY p2 = &array;
             9         //int* p3 = &array;     //這樣編譯錯誤
            10         
            11         printf("p1 = 0x%08X\n", p1);
            12         printf("p2 = 0x%08X\n", p2);
            13         printf("p1[2] = %d\n", p1[2]);
            14         printf("p2[2] = %d\n", p2[2]);
            15         printf("(*p2)[2] = %d\n", (*p2)[2]);
            16         
            17         getchar();
            18         return 0;
            19 }

            運行結(jié)果可能是:

            p1 = 0x0022FDF8
            p2 
            = 0x0022FDF8
            p1
            [2] = 2
            p2
            [2] = 2294040
            (*p2)
            [2] = 2



            posted on 2008-04-04 20:06 cuigang 閱讀(10402) 評論(21)  編輯 收藏 引用 所屬分類: C/C++

            評論

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            不錯,受教了。
            2008-04-05 10:36 | lovedday

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            我一直以為array就等于&array,因為我記得對數(shù)組名進行&操作后的值與數(shù)組名一樣,于是自以為array等于&array,看來我還沒有理解其中的含義.
            謝謝.
            2008-04-05 10:42 | passerby

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            地址一樣,解釋方式不一樣.
            2008-04-05 14:07 | hi

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            ....把數(shù)組名當個指針不就很好理解了嗎
            2008-04-05 14:50 | Santa

            # re: 對數(shù)組名取地址是什么?[未登錄]  回復(fù)  更多評論   

            @Santa
            僅僅認為數(shù)組名是個指針,是不行的。首先 array 不是一個變量,對它取地址是件很費解的事情,它并不在內(nèi)存中,可能只是一個鏈接時常量。另外,即便 array 是一個變量,存在內(nèi)存中,它的地址等于自己的值,恐怕更加頭大。
            2008-04-05 19:28 | cuigang

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            1 int array[100];
            2
            3 memset(array, 0, sizeof(array));
            4 memset(&array, 0, sizeof(array));

            第3行和第4行其實是一樣的,只是不同的寫法而已。array OR &array,編譯器取的都是數(shù)組首地址(即數(shù)組名隱式轉(zhuǎn)換成pointer類型)。而不能說array 和 &array是兩種不能的類型,數(shù)組名本來就不是一種類型。

            下面一個相似的例子:
            #include <cstdio>

            void func(char* str)
            {
            puts(str);
            }

            int main(void)
            {
            func("Normal called..");
            (*********************************func)("No ploblem called by this way.");//typeof(func)?
            (*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&*&func)("No ploblem, Just a joke.");//typeof(func)?
            return 0;
            }

            =================================================
            D:\workspace\test>g++ test0b.cc -o test0b.exe

            D:\workspace\test>test0b
            Normal called..
            No ploblem called by this way.
            No ploblem, Just a joke.

            個人意見,高手勿笑。
            2008-04-06 16:17 | Xshl5

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            非一般人啊??!@Xshl5
            2008-04-07 22:50 | 超人

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            @Xshl5
            @超人

            關(guān)于函數(shù)類型和數(shù)組類型的問題雖然表現(xiàn)不一致,但是原因是一樣的,因為說明比較長,我重新開一文說明:

            http://www.shnenglu.com/cuigang/archive/2008/04/07/46464.html

            至于數(shù)組名和函數(shù)名確實分別是數(shù)組類型和函數(shù)類型的問題,你可以去看C++ 標準 ISO14882。




            2008-04-07 22:53 | cuigang

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            這個其實是C語言的特性.
            C語言中設(shè)計的參數(shù)傳遞全部是值傳遞,除非你聲明了&引用傳遞,他才會傳遞至.
            但是又一個例外就是數(shù)組傳遞的時候是只傳首地址的,當時設(shè)計的時候考慮了性能問題.array和&array表示的效果是一樣的.
            這個在Poniters On C里面有詳細的描述.
            2008-04-09 01:12 | egmkang

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            @Xshl5
            你可能說的是狹義的“類型”,只包括int、char、float、double等。
            照這個說法 int* 都不能算是一種類型,因為:
            int* x, y;
            該聲明實際上得到整型指針變量x和整型變量y,如果int*算是一種類型的話,那么y也應(yīng)該被聲明為int*型。所以一般這樣寫:
            int *x, y;
            或者干脆分開:
            int* x;
            int y;

            在賦值語句中,不管等號左邊還是等號右邊的表達式都是有類型的。要想賦值語句合法,等號右邊的類型必須跟等號左邊的類型相同(強制轉(zhuǎn)換后相同也算),或者能夠隱式轉(zhuǎn)換到左邊。

            int array[100] = {0, 1, 2}; //array的類型是 int [100]
            int* p1 = array; //int[100]可以隱式轉(zhuǎn)換到 int*
            int* p2 = &array; //錯誤!無法從“int (*)[100]”轉(zhuǎn)換為“int *”

            實際上array返回的是array[0]的地址,我們可以用*(array + 1)來獲得array[1]的值,不用*array++是因為array為右值。

            由于int[100]能隱式轉(zhuǎn)換到int*,某些書籍在未經(jīng)考查的情況下直接說數(shù)組名array就是int*類型的,array的類型跟p1類型是一樣的。照這種說法,&array就應(yīng)該是int**類型的。但是:
            int** p3 = &p1; //正確!&p1為int**類型。
            int** p4 = &array; //錯誤!無法從“int (*)[100]”轉(zhuǎn)換為“int **”

            為了跟&array的類型匹配,LZ自己定義了一個類型:
            typedef int (*ARRAY)[100];
            ARRAY p2 = &array;

            這種做法非常聰明,學(xué)習(xí)了。
            2008-04-10 00:12 | ww

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            @ww
            類型應(yīng)該是指:在使用前有明確定義的類型,包括C++內(nèi)置類型,標準庫類型,用戶自定義類型。

            int* x,y; //對象(變量)x,y是指針類型(int *)
            int array[100]; //int[100]是編譯器為了方便而顯示的“偽”類型,當數(shù)組名被使用時,編譯器給他轉(zhuǎn)換成適當?shù)念愋汀?
            *array=0; //數(shù)組名隱式轉(zhuǎn)換成指針類型(int*)
            *(array+1)=sizeof(array); //后一個數(shù)組名不再轉(zhuǎn)換成指針類型,array[1]=400

            typedef int[100] array; //error,int[100] not a typename
            typedef int array[100]; //ok
            array arr;
            所以個人比較傾向于數(shù)組名(而不是int[100])隱式轉(zhuǎn)換成指針類型(int*),在很多時候數(shù)組名跟指針是等價的。
            int array[100]; //當然為了理解方便,去掉變量名int[100]可以看作是array的類型。在char *strncpy(char *,const char *,size_t)中,可以把函數(shù)名strncpy的類型看作是char* (char *,const char *,size_t),事實上,編譯器在報錯時就是這樣顯示的。注意char* (char *,const char *,size_t)是函數(shù)返回char*指針,和char (*)(char *,const char *,size_t)指向函數(shù)的指針,該函數(shù)返回char 不一樣。
            2008-04-10 17:00 | Xshl5

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            @Xshl5
            請注意,int* x, y;實際上聲明了一個 int* 類型的變量 x 和一個 int 類型的變量 y ,而不是兩個 int* 類型的變量。試著編譯以下語句:
            int z = 10;
            int* x, y;
            x = &z; //正確!x為 int* 類型
            y = z; //正確!y為 int 類型
            x = y; //錯誤!無法從 int 轉(zhuǎn)換為 int*
            y = &z; //錯誤!無法從 int* 轉(zhuǎn)換為 int
            2008-04-10 18:20 | ww

            # re: 對數(shù)組名取地址是什么?[未登錄]  回復(fù)  更多評論   

            @Xshl5
            關(guān)于“類型”。

            我們知道對于以下函數(shù)原型:
            void fn(int x, double y);
            可以省略掉參數(shù)名,僅用參數(shù)類型來做函數(shù)聲明,即:
            void fn(int, double);

            而以下代碼是能通過編譯的:
            void Print(int [2], int); //注意此處用int[2]作為參數(shù)類型!

            void _tmain(void)
            {
            int array[2] = {1, 2};
            Print(array, 2);
            }

            void Print(int iArray[2], int x)
            {
            //do something
            }//end of code

            是否可以說明編譯器是認可 int[2] 類型的呢?
            2008-04-10 20:27 | iwong

            # re: 對數(shù)組名取地址是什么?[未登錄]  回復(fù)  更多評論   

            @Xshl5

            看來你還比較執(zhí)著。 int [N] 是類型是無容置疑的。建議看 ISO 14882
            2008-04-10 22:24 | cuigang

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            樓主,是否可以通過下面的代碼來說明 array和&array的類型呢?

            int array[100];
            memset(array, 0, sizeof(array));

            printf("sizeof(array)=%d, sizeof(&array)=%d\n", sizeof(array), sizeof(&array));

            2008-04-21 23:36 | perddy

            # re: 對數(shù)組名取地址是什么?[未登錄]  回復(fù)  更多評論   

            @perddy

            你想說明什么呢?
            2008-04-21 23:41 | cuigang

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            @cuigang

            由于習(xí)慣使用sizeof來判斷一個變量的類型,而實驗了一下,發(fā)現(xiàn)sizeof(array)和 sizeof(&array)結(jié)果一樣,都為400,所以一開始認為這兩個的類型是一樣的。
            回頭又查了下C專家編程(第9章),其中有說:
            表達式中的數(shù)組名被編譯器當作一個指向該數(shù)組第一個元素的指針。也即array為一指針類型。
            另外注釋中有提到:在使用sizeof()、&操作符取數(shù)組指針、數(shù)組為一字符串常量初始值的情況下,對數(shù)組的引用不能用指向該數(shù)組第一個元素的指針來代替。

            因此,樓主說的是正確的,array和&array類型是不一樣的。array為一個指針,而&array是指向數(shù)組int [100]的指針。下面的代碼(計算步長)也可以證明這個結(jié)論:

            printf("array=%p, array+1=%p\n", array, array+1);
            printf("&array=%p, &array+1=%p\n", &array, &array+1);

            結(jié)果為:
            array=0012FDF0, array+1=0012FDF4
            &array=0012FDF0, &array+1=0012FF80
            2008-04-22 14:30 | perddy

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            C90規(guī)定:



            Except when it is the operand of the sizeof operator or the unary & operator, or is a character string literal used to initialize an array of character type. or is a wide string literal used to initialize an array with element type compatible with wchar-t, an lvalue that has type “array of type” is converted to an expression that has type “pointer to type” that points to the initial element of the array object and is not an lvalue.



            除了作為sizeof、&及用于初始化字符數(shù)組的字符串字面量等幾種情況外,一個具有數(shù)組類型的左值表達式被轉(zhuǎn)換為指向數(shù)組首元素的右值指針。這是一個隱式轉(zhuǎn)換過程。這個條款不僅規(guī)定了首元素地址這個數(shù)值結(jié)果,還規(guī)定了轉(zhuǎn)換結(jié)果的類型:元素指針。
            2012-05-20 10:26 | nux

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            謝謝了,問題得到了解決
            2012-06-30 14:19 | 大笨兔

            # re: 對數(shù)組名取地址是什么?  回復(fù)  更多評論   

            謝謝樓主的講解、分享和大家的討論!
            2015-06-07 13:15 | 孫磊磊
            国内精品久久久久影院薰衣草| 亚洲综合熟女久久久30p| 久久青青草原国产精品免费| 精品久久久久久亚洲精品| 久久综合中文字幕| 久久99热这里只频精品6| 狠狠色狠狠色综合久久| 久久综合久久综合久久综合| 久久伊人五月丁香狠狠色| 久久精品国产99国产电影网| 久久久久免费视频| 久久精品人人槡人妻人人玩AV| 久久久久久久久久免免费精品| 色欲综合久久躁天天躁蜜桃| 久久久噜噜噜久久| 国产欧美一区二区久久| 亚洲精品无码久久一线| 久久久久香蕉视频| 国产精品免费久久久久电影网| 人妻精品久久无码区| 久久天天躁狠狠躁夜夜2020一| A级毛片无码久久精品免费| 人人狠狠综合久久88成人| 色诱久久av| 国内精品久久久久久久影视麻豆 | 精品多毛少妇人妻AV免费久久 | 久久综合亚洲鲁鲁五月天| 国产精品欧美久久久久无广告| 久久精品日日躁夜夜躁欧美| 久久噜噜久久久精品66| 久久se这里只有精品| 日本福利片国产午夜久久| 97久久香蕉国产线看观看| 婷婷久久久亚洲欧洲日产国码AV | 久久99国产精一区二区三区| 久久精品中文字幕无码绿巨人| 日韩美女18网站久久精品| 亚洲欧美日韩精品久久亚洲区| 四虎影视久久久免费观看| 日韩精品无码久久一区二区三| 久久青青草原精品国产软件 |