C/C++靜態(tài)數(shù)組與動態(tài)數(shù)組的區(qū)別
簡介
以下三行代碼有什么區(qū)別?
int?a[10]; int?*a?=?(int*)malloc(sizeof(int)*10); int?*a?=?new?int[10];
第一行代碼定義a為包含10個int類型元素的整形數(shù)組。 第二行和第三行分別使用的是C和C++來定義動態(tài)數(shù)組,他們的結(jié)果是相同的。a都可以表示為一個動態(tài)數(shù)組。
我們可以使用a[1]來取數(shù)組a的第1個元素。那他們有什么區(qū)別呢?
解釋
我們從指針開始說起。所謂指針,就是用來存放內(nèi)存地址的一個變量,首先,指針是個變量;其次,指針存放的是內(nèi)存地址。
指針的定義中包含了一個重要的說明:指針中存放的內(nèi)存地址處的內(nèi)容應該如何解析。例如:int *a;
說明a
是一個指針,他存放的地址處的數(shù)據(jù)被解析為一系列連續(xù)的int
型數(shù)據(jù)。
那int a[10]
,可以說明a
是一個指針么?其實,這樣說是不準確的。a
其實本身就是一個內(nèi)存地址。只是我們在實際運行之前并不知道他真正代表哪個地址。就像宏在編譯的時候被替換成宏定義的內(nèi)容,靜態(tài)數(shù)組a
實際上在執(zhí)行的時候是被替換成真實的內(nèi)存地址的。也就是說a
已經(jīng)是內(nèi)存地址了,不是變量。
那么對于int a[10]
和 int *b = new int[10]
,a[2]
和b[2]
有什么區(qū)別呢?
a[2]
,b[2]
在賦值符號=
的右端的時候。此時表示取a[2]
和b[2]
的值。對于a[2]
,首先a
代表的是個內(nèi)存地址,在這個內(nèi)存地址處偏移sizeof(int)*2
個字節(jié),取連續(xù)sizeof(int)
個字節(jié),并將其解析為int
類型的數(shù)。對于b[2]
,首先b
是一個指針,其值是一個內(nèi)存地址,首先b
這個變量在內(nèi)存中的地址被找到,然后取連續(xù)的sizeof(int*)
個字節(jié),解析為一個內(nèi)存地址,然后在這個地址處偏移sizeof(int)*2
個字節(jié),取連續(xù)sizeof(int)
個字節(jié),并將其解析為int
類型的數(shù)。a[2]
,b[2]
在賦值符號=
的左端的時候。表示向相應的內(nèi)存位置賦值。賦值符號右端的表達式的結(jié)果不管是什么類型的值,都會被默認強制類型轉(zhuǎn)化為int
型數(shù)據(jù)。對于a[2]
,a
代表的是個內(nèi)存地址,在這個內(nèi)存地址處偏移sizeof(int)*2
個字節(jié),向連續(xù)的sizeof(int)
個字節(jié)內(nèi)存中寫入賦值號右端的結(jié)果。對于b[2]
,首先b
這個變量在內(nèi)存中的地址被找到,然后取連續(xù)的sizeof(int*)
個字節(jié),解析為一個內(nèi)存地址,然后在這個地址處偏移sizeof(int)*2
個字節(jié)之后,向連續(xù)的sizeof(int)
個字節(jié)內(nèi)存中寫入賦值號右端的結(jié)果。
實驗
試驗下前面討論的內(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
的值是相同的,說明a
已經(jīng)是地址了,取地址后還是原先的地址,所以兩次地址是一樣的。
后兩次輸出結(jié)果不同,是應為b
是一個指針,是變量,變量的地址與他存儲的內(nèi)存地址是不同的。