C/C++靜態(tài)數(shù)組與動(dòng)態(tài)數(shù)組的區(qū)別
掃描二維碼
隨時(shí)隨地手機(jī)看文章
簡(jiǎn)介
以下三行代碼有什么區(qū)別?
int?a[10]; int?*a?=?(int*)malloc(sizeof(int)*10); int?*a?=?new?int[10];
第一行代碼定義a為包含10個(gè)int類(lèi)型元素的整形數(shù)組。 第二行和第三行分別使用的是C和C++來(lái)定義動(dòng)態(tài)數(shù)組,他們的結(jié)果是相同的。a都可以表示為一個(gè)動(dòng)態(tài)數(shù)組。
我們可以使用a[1]來(lái)取數(shù)組a的第1個(gè)元素。那他們有什么區(qū)別呢?
解釋
我們從指針開(kāi)始說(shuō)起。所謂指針,就是用來(lái)存放內(nèi)存地址的一個(gè)變量,首先,指針是個(gè)變量;其次,指針存放的是內(nèi)存地址。
指針的定義中包含了一個(gè)重要的說(shuō)明:指針中存放的內(nèi)存地址處的內(nèi)容應(yīng)該如何解析。例如:int *a; 說(shuō)明a是一個(gè)指針,他存放的地址處的數(shù)據(jù)被解析為一系列連續(xù)的int型數(shù)據(jù)。
那int a[10],可以說(shuō)明a是一個(gè)指針么?其實(shí),這樣說(shuō)是不準(zhǔn)確的。a其實(shí)本身就是一個(gè)內(nèi)存地址。只是我們?cè)趯?shí)際運(yùn)行之前并不知道他真正代表哪個(gè)地址。就像宏在編譯的時(shí)候被替換成宏定義的內(nèi)容,靜態(tài)數(shù)組a實(shí)際上在執(zhí)行的時(shí)候是被替換成真實(shí)的內(nèi)存地址的。也就是說(shuō)a已經(jīng)是內(nèi)存地址了,不是變量。
那么對(duì)于int a[10] 和 int *b = new int[10],a[2]和b[2]有什么區(qū)別呢?
a[2],b[2]在賦值符號(hào)=的右端的時(shí)候。此時(shí)表示取a[2]和b[2]的值。對(duì)于a[2],首先a代表的是個(gè)內(nèi)存地址,在這個(gè)內(nèi)存地址處偏移sizeof(int)*2個(gè)字節(jié),取連續(xù)sizeof(int)個(gè)字節(jié),并將其解析為int類(lèi)型的數(shù)。對(duì)于b[2],首先b是一個(gè)指針,其值是一個(gè)內(nèi)存地址,首先b這個(gè)變量在內(nèi)存中的地址被找到,然后取連續(xù)的sizeof(int*)個(gè)字節(jié),解析為一個(gè)內(nèi)存地址,然后在這個(gè)地址處偏移sizeof(int)*2個(gè)字節(jié),取連續(xù)sizeof(int)個(gè)字節(jié),并將其解析為int類(lèi)型的數(shù)。a[2],b[2]在賦值符號(hào)=的左端的時(shí)候。表示向相應(yīng)的內(nèi)存位置賦值。賦值符號(hào)右端的表達(dá)式的結(jié)果不管是什么類(lèi)型的值,都會(huì)被默認(rèn)強(qiáng)制類(lèi)型轉(zhuǎn)化為int型數(shù)據(jù)。對(duì)于a[2],a代表的是個(gè)內(nèi)存地址,在這個(gè)內(nèi)存地址處偏移sizeof(int)*2個(gè)字節(jié),向連續(xù)的sizeof(int)個(gè)字節(jié)內(nèi)存中寫(xiě)入賦值號(hào)右端的結(jié)果。對(duì)于b[2],首先b這個(gè)變量在內(nèi)存中的地址被找到,然后取連續(xù)的sizeof(int*)個(gè)字節(jié),解析為一個(gè)內(nèi)存地址,然后在這個(gè)地址處偏移sizeof(int)*2個(gè)字節(jié)之后,向連續(xù)的sizeof(int)個(gè)字節(jié)內(nèi)存中寫(xiě)入賦值號(hào)右端的結(jié)果。
實(shí)驗(yàn)
試驗(yàn)下前面討論的內(nèi)容:
#include#include#includeint?main(int?argc,?char**?argv){
????int?a[10];
????printf("%.16Xn",?(uint64_t)(&a));
????printf("%.16Xn",?(uint64_t)(a));
????int?*b?=?new?int[10];
????printf("%.16Xn",?(uint64_t)(&b));
????printf("%.16Xn",?(uint64_t)(b));
????delete[]?b;?b?=?NULL;
????return?EXIT_SUCCESS;
}某次執(zhí)行結(jié)果:
000000003093F838 000000003093F838 000000003093F878 00000000309A8D90
根據(jù)輸出結(jié)果,可以推斷,a和&a的值是相同的,說(shuō)明a已經(jīng)是地址了,取地址后還是原先的地址,所以?xún)纱蔚刂肥且粯拥摹?/p>
后兩次輸出結(jié)果不同,是應(yīng)為b是一個(gè)指針,是變量,變量的地址與他存儲(chǔ)的內(nèi)存地址是不同的。





