關於數組和指針

1. 先來看幾個例子,你能說出這幾個表示方法的意義和不同點嗎?

a. int *p[10]  //10個指向int類型的指針,數組裏面的類型是int *,這個方式叫指針數組,可以轉化成二級指針;
b. int (*p)[10]  //指向一個有10個int成員的數組,其實也就是一個2維數組p[][10],確定的第二維的長度是10,數組裏面對象的類型是int,這種方式叫數組指針,可以轉化成二維數組;
c. int (*p)(int)  //指向一個形參表爲1個int類型,返回值爲int的函數指針

d. int *p(int a) //這個是一個指針函數,函數返回的是一個地址,使用在在返回數組某一個元素的地址上;
e. int (*p[10])(int) //指針數組指針函數的結合:定義一個指針數組,數組中每個成員均是int (*p)(int) 的指針;

這幾種表示方法經常出現在程序中,很容易使人混淆,需要注意的是:C語言經常用函數指針的方式實現面相對象的功能,在動態調用過程中,把實現不同功能的類似函數通過函數指針的形式賦值給真正調用的函數,來實現面相對象的過程。

 

2. 多維數組和指針

例如:整型二維數組a[3][4]:

  0  1  2  3
  4  5  6  7
  8  9  10  11

設數組a的首地址爲1000,C語言允許把一個二維數組分解爲多個一維數組來處理。因此數組a可分解爲三個一維數組,即a[0],a[1],a[2]。每一個一維數組又含有四個元素。例如a[0]數組,含有a[0][0],a[0][1],a[0][2],a[0][3]四個元素。
數組及數組元素的地址表示如下:a是二維數組名,也是二維數組0行的首地址,等於1000。a[0]是第一個一維數組的數組名和首地址,因此也爲1000,*(a+0)或*a是與a[0]等效的, 它表示一維數組a[0]的0號元素的首地址,也爲1000,&a[0][0]是二維數組a的0行0列元素首地址,同樣是1000。
因此,a,a[0],*(a+0),*a,&a[0][0]是相同的。同理,a+1是二維數組1行的首地址,等於1008。a[1]是第二個一維數組的數組名和首地址,因此也爲1008。&a[1][0]是二維數組a的1行0列元素地址,也是1008。因此a+1,a[1],*(a+1),&a[1][0]是等同的。由此可得出:a+i,a[i],*(a+i),&a[i][0]是等同的。此外,&a[i]和a[i]也是等同的。因爲在二維數組中不能把&a[i]理解爲元素a[i]的地址,不存在元素a[i]。C語言規定,它是一種地址計算方法,表示數組a第i行首地址。由此,我們得出:a[i],&a[i],*(a+i)和a+i也都是等同的。另外,a[0]也可以看成是a[0]+0是一維數組a[0]的0號元素的首地址,而a[0]+1則是a[0]的1號元素首地址,由此可得出a[i]+j則是一維數組a[i]的j號元素首地址,它等於&a[i][j]。由a [i]=*(a+i)得a[i]+j=*(a+i)+j,由於*(a+i)+j是二維數組a的i行j列元素的首地址。該元素的值等於*(*(a+i)+ j)。

試一下以下程序段

#define PF "%d,%d,%d,%d,%d,\n"
int main(void)

{
    static int a[3][4]={0,1,2,3,4,5,6,7,8,9,10,11};
    printf(PF,a,*a,a[0],&a[0],&a[0][0]);
    printf(PF,a+1,*(a+1),a[1],&a[1],&a[1][0]);
    printf(PF,a+2,*(a+2),a[2],&a[2],&a[2][0]);
    printf("%d,%d\n",a[1]+1,*(a+1)+1);
    printf("%d,%d\n",*(a[1]+1),*(*(a+1)+1));

    return 0;
}

運行結果如下:(注意,是打印的10進制數)


其中數組地址信息如下:


說明:0x003a7038的十進制爲3829816

3. 二級指針做參數

通過下面的例子來了解二級指針做函數參數時,主函數對此函數是怎麼調用的。

#include "stdio.h"

typedef struct tst
{
 int x;
}tst_t;

typedef struct student
{
     int a;
     tst_t *s;
     int c;
}stu_t;

void fun(stu_t **p)
{
     int i = 0;
     for (i = 0; i < 4; i++)
     {
         if ((*(p + i))==NULL)
         {
             printf("yes\n");
         }
         else
         {
             printf("no!\n");
             if (((*(p + i))->s->x) != 0) 
             {
                  printf("no zero\n");
             }
             else
             {
                  printf("zero\n");
             }
         }
     }
}
int _tmain(int argc, _TCHAR* argv[])
{
     stu_t *tmp[4];
     stu_t t;
     tst_t p[4];
     p[0].x = 1;
     p[1].x = 0;
     p[2].x = 0;
     p[3].x = 0;
     tmp[0] = &t;
     tmp[0]->s = p;
     tmp[1] = NULL;
     //tmp[1]->s = &p[1];由於tmp[1]指向了null,調用tmp[1]的指針成員時候發生中斷異常。
     tmp[2] = &t;
     tmp[2]->s = &p[2];

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章