• <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語言中數(shù)組名和指針的區(qū)別

            從編譯器角度分析C語言中數(shù)組名和指針的區(qū)別


            數(shù)組名和指針是兩個往往很容易讓人們混淆的概念,
            很多人以為數(shù)組名就是一個指針,
            也有很多人知道數(shù)組名不同于指針但是僅知道數(shù)組名的值不能像指針一樣改變


            例如你可以寫出下面這樣的代碼:
            int *p;
            p++;
            卻不能寫這樣的代碼:
            int a[];
            a++;


            那么數(shù)組名跟指針之間到底有什么區(qū)別呢?
            第一,在聲明上,只有作為函數(shù)參數(shù)的數(shù)組名編譯器總是轉(zhuǎn)化成指針,
            其他情況下,數(shù)組名就是數(shù)組名,指針就是指針,二者不能混淆,
            你不能在一個文件中定義一個數(shù)組,而在另一個文件中把它聲明成一個指針。
            char a[];    //定義一個數(shù)組a
            extern char* a;    //在另一個文件中將a聲明成一個指針


            在編譯器中,符號表用來存放C語言中有關(guān)標(biāo)識符的屬性信息,這些信息集中反應(yīng)了標(biāo)識符的特征屬性。
            等到詞法分析到代碼聲稱的各個階段的時候,編譯器需要根據(jù)源代碼提出的要求,從表中獲取不同標(biāo)識符的不同屬性。
            值得注意的是,數(shù)組標(biāo)識符的屬性和指針是完全不同的。
            因此,在一個文件中定義一個數(shù)組的時候,編譯器會把它記錄在符號表中,而在編譯器分析另一個文件中的聲明時,通過符號的語義檢查發(fā)現(xiàn)不一致。
            也許這樣的聲明可以編譯通過(在gcc下通過了),但是把一個數(shù)組名當(dāng)成一個指針來使用,別指望它能運(yùn)行起來。


            第二,指針是一個變量,而數(shù)組名不是。
            數(shù)組名是數(shù)組的首地址,它本身就是一個地址,對應(yīng)到匯編語言級別就是一個常量,一個固定的數(shù)(地址)。
            因此數(shù)組名不能進(jìn)行++,--等運(yùn)算。
            在大多數(shù)編譯器中,對數(shù)組的引用a[i]總是被編譯器改寫成*(a+i)的格式。
            也就是說,編譯器每遇到a[i],都會把它當(dāng)作*(a+i)來處理。
            我們都知道,*addr表示內(nèi)存中(addr)的位置存儲的值,
            比如*0x8048000就表示地址為0x8048000的內(nèi)存中所存儲的值。
            所以a[i]就表示a的值加上i所得到的數(shù)作為一個內(nèi)存地址里面所存儲的值。


            那么a的值是什么呢 ???
            編譯器在做詞法分析和語法分析的時候,遇到一個數(shù)組的定義,就會把數(shù)組的有關(guān)信息匯集在一個叫做“內(nèi)情向量”或“信息向量”的表格中,其中的信息包括數(shù)組的類型,維數(shù),各維的上、下邊界,以及數(shù)組的首地址,然后將這個“內(nèi)情向量”相關(guān)信息存儲在符號表中。
            數(shù)組定義后位置就是固定的,因此其首地址可以在編譯階段得到。


            當(dāng)編譯器到達(dá)代碼生成的各階段時,每次遇到數(shù)組名這個標(biāo)識符,編譯器都會從符號表中取出這個數(shù)組的首地址,然后用這個地址來替代數(shù)組名,例如,假設(shè)數(shù)組a起始地址是0x8048000,則a[1]就會被編譯器轉(zhuǎn)化成*(0x8048000+1),
            因此在生成的匯編代碼中,數(shù)組名已經(jīng)完全被轉(zhuǎn)化成一個常量,一個固定的數(shù)(地址)。
            但是
            對于指針p,它是一個變量,其值存儲在地址&p中,這個值在編譯時是不能得到的。
            因為是變量,所以指針可以作為表達(dá)式中的左值操作,
            如++或--,而被認(rèn)為是常量的數(shù)組名卻不可以,正如你可以騎走一輛自行車,但是不能騎走一棵樹。


            另外,C語言把 數(shù)組下標(biāo) 改寫成 指針偏移量 的  根本原因  是
            指針偏移量是底層硬件所使用的基本模型。


            第三,對數(shù)組的引用,如a[i],或*(a+1),需要訪存一次;而對指針的引用,如*(p+1),需要訪存兩次。


            a被認(rèn)為是常數(shù),所以取*(a+1)的值只需將a所表示的常數(shù)加1,然后從得到的地址里訪存取一次即可。
            對于指針,需要先從&p這個地址里把p的值取出來,然后加1,再從得到的地址里訪存取一次,一共需要兩次訪存。


            第四,假設(shè)a是一個數(shù)組名,而p是一個指針,當(dāng)你使用 a 和 &a 時,得到值是一樣的,都是數(shù)組的起始地址。
            而使用 p 和 &p 時,得到的值是不一樣的, p 表示指針 p 所指向的地址,而 &p 表示 p 這個變量的地址。
            再假設(shè)   p = a;
            則 p 就表示數(shù)組a的起始地址,而&p是存儲數(shù)組a的起始地址的那個地址。
            這是因為編譯器把a(bǔ)當(dāng)成數(shù)組首地址,而&a當(dāng)作數(shù)組第一個元素的地址,因此得到的值是一樣的。
            另外,sizeof(a)得到的是a所表示的數(shù)組的大小,而sizeof(p)得到的是指針的大小。

            轉(zhuǎn)自:
            http://blog.chinaunix.net/uid-27004869-id-3301282.html

            posted on 2012-08-07 09:00 天下 閱讀(418) 評論(0)  編輯 收藏 引用 所屬分類: C/C++

            <2012年10月>
            30123456
            78910111213
            14151617181920
            21222324252627
            28293031123
            45678910

            導(dǎo)航

            統(tǒng)計

            常用鏈接

            留言簿(4)

            隨筆分類(378)

            隨筆檔案(329)

            鏈接

            最新隨筆

            搜索

            最新評論

            怡红院日本一道日本久久 | 国内精品久久久久久久久电影网 | 99久久国产综合精品网成人影院| 成人国内精品久久久久影院| 国产精品一区二区久久精品无码 | 国产精品99久久久久久董美香| 性欧美大战久久久久久久| 久久婷婷色香五月综合激情| 精品无码久久久久国产动漫3d| 一级做a爱片久久毛片| 精品国产乱码久久久久久呢| 国产成人精品久久一区二区三区av | 伊人久久精品无码av一区| 99久久精品费精品国产一区二区| 亚洲欧美久久久久9999| 一本伊大人香蕉久久网手机| 久久久久久久人妻无码中文字幕爆| 国产午夜精品理论片久久| 性高湖久久久久久久久| 天天做夜夜做久久做狠狠| 2022年国产精品久久久久| 欧美日韩中文字幕久久久不卡| 久久精品这里热有精品| 久久亚洲AV成人无码电影| 久久久久久久精品成人热色戒| 久久99久久无码毛片一区二区| 91精品国产91久久久久福利| 久久久久亚洲AV成人片| 久久青青国产| 亚洲国产成人精品女人久久久| 久久久久国产一区二区三区| 国产视频久久| 色偷偷88欧美精品久久久| 久久青青国产| 囯产精品久久久久久久久蜜桃 | 亚洲中文字幕无码久久精品1| 色欲综合久久躁天天躁| 久久综合色老色| 久久久久久久综合狠狠综合| 亚洲另类欧美综合久久图片区| 亚洲&#228;v永久无码精品天堂久久|