在 C 語(yǔ)言中,變量的地址往往都是編譯系統(tǒng)自動(dòng)分配的,對(duì)我們用戶來(lái)說(shuō),我們是不知道某個(gè)變量的具體地址的。所以我們定義一個(gè)指針變量 p,把普通變量 a 的地址直接送給指針變量 p 就是 p = &a;這樣的寫法。
對(duì)于指針變量 p 的定義和初始化,一般有兩種方式,這兩種方式,初學(xué)者很容易混淆,因此這個(gè)地方?jīng)]別的方法,就是死記硬背,記住即可。
方法 1:定義時(shí)直接進(jìn)行初始化賦值。
unsigned char a;
unsigned char *p = &a;
方法 2:定義后再進(jìn)行賦值。
unsigned char a;
unsigned char *p;
p = &a;
大家仔細(xì)看會(huì)看出來(lái)這兩種寫法的區(qū)別,它們都是正確的。我們?cè)诙x的指針變量前邊加了個(gè)*,這個(gè)*p 就代表了這個(gè) p 是個(gè)指針變量,不是個(gè)普通的變量,它是專門用來(lái)存放變量地址的。此外,我們定義*p 的時(shí)候,用了 unsigned char 來(lái)定義,這里表示的是這個(gè)指針指向的變量類型是 unsigned char 型的。
指針變量似乎比較好理解,大家也能很容易就聽明白。但是為什么很多人弄不明白指針呢?因?yàn)樵?C 語(yǔ)言中,有一些運(yùn)算和定義,他們是有區(qū)別的,很多同學(xué)就是沒弄明白它們的區(qū)別,指針就始終學(xué)不好。這里我要重點(diǎn)強(qiáng)調(diào)兩個(gè)區(qū)別,只要把這兩個(gè)區(qū)別弄明白了,起碼指針變量這部分就不是問題了。這兩個(gè)重點(diǎn)現(xiàn)在大家死記硬背,直接記住即可,靠理解有可能混淆概念。
第一個(gè)重要區(qū)別:指針變量 p 和普通變量 a 的區(qū)別。
我們定義一個(gè)變量 a,同時(shí)也可以給變量 a 賦值 a = 1,也可以賦值 a = 2。
我們定義一個(gè)指針變量 p,另外還定義了一個(gè)普通變量 a=1,普通變量 b=2,那么這個(gè)指針變量可以指向 a 的地址,也可以指向 b 的地址,可以寫成 p = &a,也可以寫成 p = &b,但就是不能寫成 p = 1 或者 p = 2 或者 p = a,這三種表達(dá)方式都是錯(cuò)的。
因此這個(gè)地方,不要看到定義*p 的時(shí)候前邊有個(gè) unsigned char 型,就錯(cuò)誤的賦值 p=1,這個(gè)只是說(shuō)明 p 指向的變量是這個(gè) unsigned char 類型的,而 p 本身,是指針變量,不可以給它賦值普通的值或者變量,后邊我們會(huì)直接把指針變量稱之為指針,大家要注意一下這個(gè)小細(xì)節(jié)。
前邊這個(gè)區(qū)別似乎比較好理解,還有第二個(gè)重要區(qū)別,一定要記清楚。
第二個(gè)重要區(qū)別:定義指針變量*p 和取值運(yùn)算*p 的區(qū)別。
“*”這個(gè)符號(hào),在我們的 C 語(yǔ)言有三個(gè)用法,第一個(gè)用法很簡(jiǎn)單,乘法操作就是用這個(gè)符號(hào),這里就不講了。
第二個(gè)用法,是定義指針變量的時(shí)候用的,比如 unsigned char *p,這個(gè)地方使用“*”代表的意思是 p 是一個(gè)指針變量,而非普通的變量。
還有第三種用法,就是取值運(yùn)算,和定義指針變量是完全兩碼事,比如:
unsigned char a = 1;
unsigned char b = 2;
unsigned char *p;
p = &a;
b = *p;
這樣兩步運(yùn)算完了之后,b 的值就成了 1 了。在這段代碼中,&a 表示取 a 這個(gè)變量的地址,把這個(gè)地址送給 p 之后,再用*p 運(yùn)算表示的是取指針變量 p 指向的地址的變量的值,又把這個(gè)值送給了 b,最終的結(jié)果相當(dāng)于 b=a。同樣是*p,放在定義的位置就是定義指針變量,放在執(zhí)行代碼中就是取值運(yùn)算。
這兩個(gè)重要區(qū)別,大家可以反復(fù)閱讀三四遍,把這兩個(gè)重要區(qū)別弄明白,指針的大門就順利的踏進(jìn)去一只腳了。至于詳細(xì)的用法,我們后邊用得多了就會(huì)慢慢熟悉起來(lái)了。