數組指針(也稱行指針)
定義 int (*p)[n];
()優先級高,首先說明p是一個指針,指向一個整型的一維數組,這個一維數組的長度是n,也可以說是p的步長。也就是說執行p+1時,p要跨過n個整型數據的長度。
如要將二維數組賦給一指針,應這樣賦值:
int a[3][4];
int (*p)[4]; //該語句是定義一個數組指針,指向含4個元素的一維數組。
p=a; //將該二維數組的首地址賦給p,也就是a[0]或&a[0][0]
p++; //該語句執行過後,也就是p=p+1;p跨過行a[0][]指向了行a[1][]
所以數組指針也稱指向一維數組的指針,亦稱行指針。
指針數組
定義 int *p[n];
[]優先級高,先與p結合成為一個數組,再由int*說明這是一個整型指針數組,它有n個指針類型的數組元素。這裡執行p+1是錯誤的,這樣賦值也是錯誤的:p=a;因為p是個不可知的表示,只存在p[0]、p[1]、p[2]...p[n-1],而且它們分別是指針變量可以用來存放變量地址。但可以這樣 *p=a; 這裡*p表示指針數組第一個元素的值,a的首地址的值。
如要將二維數組賦給一指針數組:
int *p[3];
int a[3][4];
for(i=0;i<3;i++)
p[i]=a[i];
這裡int *p[3] 表示一個一維數組內存放著三個指針變量,分別是p[0]、p[1]、p[2]
所以要分別賦值。
這樣兩者的區別就豁然開朗了,數組指針只是一個指針變量,似乎是C語言裡專門用來指向二維數組的,它占有內存中一個指針的存儲空間。指針數組是多個指針變量,以數組形式存在內存當中,占有多個指針的存儲空間。
還需要說明的一點就是,同時用來指向二維數組時,其引用和用數組名引用都是一樣的。
比如要表示數組中i行j列一個元素:
*(p[i]+j)、*(*(p+i)+j)、(*(p+i))[j]、p[i][j]
int a[3][4]這個無需多說,就是一個二維數組。
int (*p)[4]就相當於int p[][4],它就是一個二維數組的指針,可以指向一個第二維度為4的二維數組。而a就是這樣的數組,因而下面是合法的。
p=a;
int *p[3]是指針數組。說白了,就是定義了三個指針,分別為p[0],p[1],p[2]。可以將他們單獨拿來使用。
int a1,a2,a3;
p[0]=&a1;
p[1]=&a2;
p[2]=&a3;
1 這兩種形式沒有任何區別,定義的都是char* 類型的指針,不過有的書上建議采用*和變量相靠近的寫法,這樣可以避免連續聲明時的錯誤,例如:char *ptr,a;可以明顯知道一個是指針的變量一個是char類型的變量。
2 二維數組實際可以聲明成指向指針的指針。這樣在接受形參的時候,就可以用一個雙重指針來接受。
void f(int **a) //怎麼寫?
{
}
int a[100][100];
f(a);
3 返回值是一個指針就可以實現了,不過返回的地址一定要是在堆空間申請的空間,或者是在靜態區的空間。不要返回在函數棧中的地址,以防止在退出函數時,棧空間釋放,成為幽靈內存。
char* fun(){
char* ptr ="abc";
return ptr;
}