發信人: ax.bbs@bbs.ee.nthu.edu.tw. (athena), 信區: test
標  題: 星星流講座 0034
發信站: ☆清華電機☆ (Thu Jul  6 13:57:40 1995)
轉信站: star

星星流講座 0034         C 語言教室

第 5 講 之 10           陣列與指標
                        Topic: Differences between pointers and arrays


我們前面的程式,如果把 printf 改一改,會得到三種不同的結果:

1. printf ("%s %s\n", lineptr[0], *(lineptr + 1));

        執行結果:
        I am a student I am a teacher

2. printf ("%s %s\n", lineptr[0], *(++lineptr));

        wrong type argument to increment

3. printf ("%s\n", ++*lineptr);

        執行結果:
         am a student

先看第三個,++*lineptr 的意思是把 *lineptr 加 1。那 *lineptr 又是
什麼東西呢? *lineptr 就是指向 I am a student 這個字串的指標,你把
這個指標加 1,這個時候字串開頭的字元 I 自然被「吃掉」了,所以執行
結果變成只剩 am a student 被印出來。

比較有意思的問題是,第一個是把 lineptr 加 1 之後,印出它所指向的字
串,也就是第一個相當於

        printf ("%s %s\n", lineptr[0], lineptr[1]);

但是第二個 printf 也是把 lineptr 加 1,為什麼 compiler 會給我們錯誤
訊息呢?這是因為陣列雖然可以用指標的觀念來看待,但是對編譯器而言,
陣列與指標的資料型態 (data type) 並不相同。由於陣列的資料型態不可以
使用 ++ 這個運算元 (你想 ++ 陣列的那一個元素呢?),所以這個程式會在
編譯時發生錯誤。請你看看下面的程式比較一下:

/* pa.c */
#include <stdio.h>
#include <string.h>

void foo (char **lineptr);
void main (void)
{
    char s1[20], s2[20], s3[20];
    char *lineptr[3];

    strcpy (s1, "I am a student");
    strcpy (s2, "I am a teacher");
    strcpy (s3, "I am an audiance");

    lineptr[0] = s1;
    lineptr[1] = s2;
    lineptr[2] = s3;

    foo (lineptr);
}

void foo (char **lineptr)
{
    printf ("%s\n", *++lineptr);
}


執行結果:

I am a teacher

這裡為什麼就可以這樣兒做了呢?請看 foo 的參數:

        void foo (char **lineptr)

陣列在傳給 foo 這個函數之前,已經經由隱含的型態轉換轉變成
char ** 型態了,而非原來宣告的 * char [] 型態,所以可以安心地
使用 ++ 這個運算元。

--
本文原作者為徐振家,原作刊載於星星神教總壇 ☆清華電機☆ test 板。
你可以以電子文件的形式將本文自由流傳於台灣學術網路,但必須包含此版權聲明。
原作者依中華民國著作權法之規定,享有本文之著作權,請勿抄襲以免觸法。
未經授權任何人不得以任何形式對本文做任何修改及商業上之應用。
其他網路的轉載或其他用途的應用,請先知會作者,並取得其同意。
對本文有任何疑問或意見請 mail 給 ax.bbs@bbs.ee.nthu.edu.tw,謝謝。