龍貓公交車 http://home.cnblogs.com/233971/ 在我前一篇《c語言指標與數組》http://www.cnblogs.com/lua5/archive/2010/12
代碼
void clear_string(char *str)
{
str = NULL;
}
void get_string(char *str)
{
char ch;
int i;
for(i = 0; i < 3; i++) {
ch = getchar();
if (ch == '\n')
return;
str[i] = ch;
}
}
int main(void)
{
char string[4] = {'\0'};
get_string(string);
printf("Before clear, string:%s\n", string);
clear_string(string);
printf("After clear, string:%s\n", string);
return 0;
}
首先說這段代碼的問題,為何clear_string不生效,龍貓的理解其實是不對的。對於main函數中的string變數以及clear_string中的str,它們指向同一個地址,由於所指定的類型也是一樣的,基本可以認為它們就是一回事,只是要時刻牢記我們調用clear_string(string);,實際上是把string第一個元素起始地址作為參數傳入。
要記住一個原則:C語言中,任何函數想修改傳入的值,然後返回給調用方,一定是類似這樣的 *pOut = value;
這個原則非常非常重要。
回到clear_string()這個函數,str = NULL;是什麼效果呢?這意味著我們從這一行開始,到有效地block結束(對當前函數而言block結束就是函數結尾)或者下一個對str的賦值之前,我們把str賦值為NULL。注意我們修改的是str的值,相當於把str重新指向NULL這個地址。在這裡,str是作為函數的局部變數使用,不影響調用方傳入的值,這就好比這樣一個函數原型
void SetZero(int iVal)
{
iVal = 0;
}
任何一個稍有C語言基礎的朋友都知道,當我們這樣調用int iValCaller = 5; SetZero(iValCaller);運行以後,iValCaller的值依然是5。可以這樣認為,程式運行調用SetZero函數的時候,把參數iVal在程式運行棧上做了一份拷貝(copy)傳遞給函數,函數SetZero對拷貝(copy)做的修改不會影響到調用方變數。
如果想影響母體,只有一種方式,就是把調用方變數的地址傳遞給函數,這時候通過指標就可以間接操作調用方變數值了,這也是指標的最大功用。
知道了問題,修改就很容易了,下面的寫法才是指標型變數最常用的程式碼片段:
void clear_string(char *str){ *str = NULL; }
如果我想修改str,比如某個函數負責申請一塊記憶體返回,這時候我們修改的是指標地址(&p)的值,那麼就要傳入需要指標的指標。
void AllocBuffer(char** ppBuf, int size)
{
*ppBuf = malloc(size);
}
注意,其實質還是*p = value;這種模式。