1. 程式人生 > >int *ptr=(int *)(&a+1)

int *ptr=(int *)(&a+1)

int *ptr=(int *)(&a+1)(面試題),有需要的朋友可以參考下。




【問題】:請寫出以下程式的輸出結果。



int main()
{
	int a[5]={1,2,3,4,5};
	int *ptr=(int *)(&a+1);
	printf("%d,%d\n",*(a+1),*(ptr-1));
}
【分析】:此題需要理解指標和陣列的內在含義。

【結果】:2,5


【理論知識】:我們首先需要搞明白a,&a.


a既是資料名,又是指向陣列第一個元素的指標。


sizeof(a)=20, 此時a的型別為int[5]陣列。


sizeof(*a)=4,因為有取值符*,表示把a當成一個指標(int*),而a指向陣列的首地址,


即a=&(a[0]),即sizeof(*a)=sizeof(*&(a[0]))=sizeof(a[0])=sizeof(int)=4。


*(a+1)中把a當成一個指標,a+1=a+sizeof(int),a+1指向a的下一個整形地址既&a[1]。


因此*(a+1)=*(&a[1])=a[1]=2。


(&a+1)先取變數a的地址,並根據a的地址獲得下一個與a同類型的相鄰地址。根據前面所說的a的型別為int[5]陣列。


&a+1=&a+sizeof(5*int),因此&a+1指向的地址為&a[5](陣列a[5]的下一個地址)。


(int*)(&a+1)把這個相鄰地址顯式型別轉換為int型別的地址int*ptr=(int*)(&a+1);


所以ptr指向&a[5],並且ptr是一個int型別的指標。


ptr-1=ptr-sizeof(int),故ptr-1指向&a[4]。因此,*(ptr-1)的值即為a[4]=5。


【個人理解】:


a[5]即可以看成是一個一維陣列,也可以看成是一個只有一行的二維陣列a[1][5]。


所以a指向一維陣列的首地址,即a=&a[0],a+1指向&a[1]。每次加的地址長度為sizeof(int).


而&a指向二維陣列的首地址,即&a=&a[0][5],&a+1指向&a[1][5]。每次加的地址長度為sizeof(5*int).