當(dāng)前位置:首頁 > 嵌入式培訓(xùn) > 嵌入式學(xué)習(xí) > 講師博文 > 野指針和空指針的兩個(gè)小點(diǎn)
大家都知道指針的學(xué)習(xí)對于c語言學(xué)習(xí)來說可謂是至關(guān)重要的,下面我們來說一下在指針中兩種比較特殊的關(guān)于指針的概念,野指針和空指針。
首先說一下空指針。void* 這不叫空指針,這叫無確切類型指針.這個(gè)指針指向一塊內(nèi)存,卻沒有告訴程序該用何種方式來解釋這片內(nèi)存.所以這種類型的指針不能直接進(jìn)行取內(nèi)容的操作.必須先轉(zhuǎn)成別的類型的指針才可以把內(nèi)容解釋出來。還有'\0',這也不是空指針?biāo)傅膬?nèi)容. '\0'是表示一個(gè)字符串的結(jié)尾而已,并不是NULL的意思。真正的空指針是說,這個(gè)指針沒有指向一塊有意義的內(nèi)存,比如說:
char* k;
這里這個(gè)k就叫空指針.我們并未讓它指向任意地點(diǎn)。又或者
char* k = NULL;
這里這個(gè)k也叫空指針,因?yàn)樗赶騈ULL 也就是0,注意是整數(shù)0,不是'\0'。一個(gè)空指針我們也無法對它進(jìn)行取內(nèi)容操作,這是初學(xué)者很容易犯得一個(gè)錯(cuò)誤,空指針只有在真正指向了一塊有意義的內(nèi)存后,我們才能對它取內(nèi)容。也就是說要這樣
k = "hello world!";
這時(shí)k指向一個(gè)字符串常亮就不是空指針了。
再來說一下野指針。"野指針"不是NULL指針,是指向"垃圾"內(nèi)存的指針。人們一般不會錯(cuò)用NULL指針,因?yàn)橛胕f語句很容易判斷。但是"野指針"是很危險(xiǎn)的,if語句對它不起作用。
"野指針"的成因主要有兩種:
(1)指針變量沒有被初始化。任何指針變量剛被創(chuàng)建時(shí)不會自動(dòng)成為NULL指針,它的缺省值是隨機(jī)的,它會亂指一氣。所以,指針變量在創(chuàng)建的同時(shí)應(yīng)當(dāng)被初始化,要么將指針設(shè)置為NULL,要么讓它指向合法的內(nèi)存。例如
char *p = NULL;
char *str = (char *) malloc(100);
(2)指針p被free或者delete之后,沒有置為NULL,讓人誤以為p是個(gè)合法的指針。
free和delete只是把指針?biāo)傅膬?nèi)存給釋放掉,但并沒有把指針本身干掉。指針p被free以后其地址仍然不變(非NULL),只是該地址對應(yīng)的內(nèi)存是垃圾,p成了"野指針"。如果此時(shí)不把p設(shè)置為NULL,會讓人誤以為p是個(gè)合法的指針。
如果程序比較長,我們有時(shí)記不住p所指的內(nèi)存是否已經(jīng)被釋放,在繼續(xù)使用p之前,通常會用語句if (p != NULL)進(jìn)行防錯(cuò)處理。很遺憾,此時(shí)if語句起不到防錯(cuò)作用,因?yàn)榧幢鉷不是NULL指針,它也不指向合法的內(nèi)存塊。
char *p = (char *) malloc(100);
strcpy(p, "hello");
free(p); // p 所指的內(nèi)存被釋放,但是p所指的地址仍然不變
…
if(p != NULL) // 沒有起到防錯(cuò)作用
{
strcpy(p, "world"); // 出錯(cuò)
}
p成為野指針
(3)指針操作超越了變量的作用范圍。這種情況讓人防不勝防,示例程序如下:
class A
{
public:
void Func(void){ cout << "Func of class A" << endl; }
};
void Test(void)
{
A *p;
{
A a;
p = &a; // 注意 a 的生命期
}
p->Func(); // p是"野指針"
}
函數(shù)Test在執(zhí)行語句p->Func()時(shí),對象a已經(jīng)消失,而p是指向a的,所以p就成了"野指針"。
希望今天討論的這兩點(diǎn)關(guān)于指針的話題能對剛剛學(xué)習(xí)指針的同學(xué)有一定的幫助。