• <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, 評(píng)論 - 137, 引用 - 0
            數(shù)據(jù)加載中……

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

            這兩天有人問(wèn)以下有什么代碼有什么不同?

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

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

             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;     //這樣編譯錯(cuò)誤
            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 }

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

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



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

            評(píng)論

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

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

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

            我一直以為array就等于&array,因?yàn)槲矣浀脤?duì)數(shù)組名進(jìn)行&操作后的值與數(shù)組名一樣,于是自以為array等于&array,看來(lái)我還沒(méi)有理解其中的含義.
            謝謝.
            2008-04-05 10:42 | passerby

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

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

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

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

            # re: 對(duì)數(shù)組名取地址是什么?[未登錄](méi)  回復(fù)  更多評(píng)論   

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

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

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

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

            下面一個(gè)相似的例子:
            #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.

            個(gè)人意見(jiàn),高手勿笑。
            2008-04-06 16:17 | Xshl5

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

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

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

            @Xshl5
            @超人

            關(guān)于函數(shù)類型和數(shù)組類型的問(wèn)題雖然表現(xiàn)不一致,但是原因是一樣的,因?yàn)檎f(shuō)明比較長(zhǎng),我重新開(kāi)一文說(shuō)明:

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

            至于數(shù)組名和函數(shù)名確實(shí)分別是數(shù)組類型和函數(shù)類型的問(wèn)題,你可以去看C++ 標(biāo)準(zhǔn) ISO14882。




            2008-04-07 22:53 | cuigang

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

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

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

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

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

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

            實(shí)際上array返回的是array[0]的地址,我們可以用*(array + 1)來(lái)獲得array[1]的值,不用*array++是因?yàn)閍rray為右值。

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

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

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

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

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

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

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

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

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

            # re: 對(duì)數(shù)組名取地址是什么?[未登錄](méi)  回復(fù)  更多評(píng)論   

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

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

            而以下代碼是能通過(guò)編譯的:
            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

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

            # re: 對(duì)數(shù)組名取地址是什么?[未登錄](méi)  回復(fù)  更多評(píng)論   

            @Xshl5

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

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

            樓主,是否可以通過(guò)下面的代碼來(lái)說(shuō)明 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: 對(duì)數(shù)組名取地址是什么?[未登錄](méi)  回復(fù)  更多評(píng)論   

            @perddy

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

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

            @cuigang

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

            因此,樓主說(shuō)的是正確的,array和&array類型是不一樣的。array為一個(gè)指針,而&array是指向數(shù)組int [100]的指針。下面的代碼(計(jì)算步長(zhǎng))也可以證明這個(gè)結(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: 對(duì)數(shù)組名取地址是什么?  回復(fù)  更多評(píng)論   

            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ù)組的字符串字面量等幾種情況外,一個(gè)具有數(shù)組類型的左值表達(dá)式被轉(zhuǎn)換為指向數(shù)組首元素的右值指針。這是一個(gè)隱式轉(zhuǎn)換過(guò)程。這個(gè)條款不僅規(guī)定了首元素地址這個(gè)數(shù)值結(jié)果,還規(guī)定了轉(zhuǎn)換結(jié)果的類型:元素指針。
            2012-05-20 10:26 | nux

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

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

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

            謝謝樓主的講解、分享和大家的討論!
            2015-06-07 13:15 | 孫磊磊
            久久99热这里只频精品6| 欧美精品丝袜久久久中文字幕| 国产精品久久久久久吹潮| 国产精品99久久免费观看| 伊人色综合久久| 狠狠色丁香婷婷久久综合五月| 亚洲国产精品无码久久一区二区 | 久久人人爽人人爽人人片AV麻豆 | 99久久做夜夜爱天天做精品| 日韩精品久久久久久久电影蜜臀| 久久se精品一区精品二区| 久久亚洲精品无码观看不卡| 色欲av伊人久久大香线蕉影院 | 欧美一区二区精品久久| 蜜桃麻豆www久久国产精品| 久久综合给合久久狠狠狠97色| 国产激情久久久久影院老熟女免费 | 久久婷婷五月综合色高清| 精品久久人人妻人人做精品| 无遮挡粉嫩小泬久久久久久久| 激情综合色综合久久综合| 日韩AV无码久久一区二区 | 国产精品美女久久久m| 无码乱码观看精品久久| 青青草原综合久久大伊人精品| 久久99热这里只有精品国产| 韩国三级中文字幕hd久久精品| 久久婷婷五月综合97色一本一本 | 久久精品国产亚洲AV大全| 欧美与黑人午夜性猛交久久久 | 韩国无遮挡三级久久| 伊人久久无码中文字幕| 四虎影视久久久免费观看| 亚洲国产精品久久66| 久久99精品久久久久子伦| 久久人妻无码中文字幕| 久久精品国产72国产精福利| 久久精品视频网| 精品九九久久国内精品| 精品免费久久久久久久| 亚洲中文久久精品无码|