B是一個指針變量,其中存放著C的地址,但是B也要占空間的啊,所以B也有地址,B的起始地址是0x00000004,但是B內(nèi)存中存放的是C的地址,所以B里面的內(nèi)容就是0x00000008。 那么到此為止都比較好理解: B= 0x00000008; //B的內(nèi)容 *B = '一段內(nèi)容'; //B解引用,也就是B指針指向的C的值&B = 0x00000004; //B取地址,B的地址是0x00000004 那么,再來看A:A是二級指針變量,其中存放著B的地址0x00000004,A也有地址,是0x00000000; *A = B= 0x00000008; //A解引用也就是B的內(nèi)容 **A = *B = '一段內(nèi)容'; //B解引用,也就是B指針指向的C的值A(chǔ) = &B = 0x00000004; //A存的是B的地址,B的地址是0x00000004&A = 0x00000000; //A取地址 二、使用 二級指針作為函數(shù)參數(shù)的作用:在函數(shù)外部定義一個指針p,在函數(shù)內(nèi)給指針賦值,函數(shù)結(jié)束后對指針p生效,那么我們就需要二級指針。 看看下面一段代碼:有兩個變量a,b,指針q,q指向a,我們想讓q指向b,在函數(shù)里面實現(xiàn)。 1.先看看一級指針的實現(xiàn) #include 這么寫有什么問題?為什么*q不等于100?我們看一下輸出便知: &a=0032F000,&b=0032F004,&q=0032F228 我們看輸出: note:1->a,b,q都有一個地址. note:2->q指向a. note:3->我們發(fā)現(xiàn)參數(shù)p的地址變了,跟q不一樣了,是的參數(shù)傳遞是制作了一個副本,也就是p和q不是同一個指針,但是指向的地址0x0032F000(a的地址)還是不變的. note:4->p重新指向b. note:5->退出函數(shù),p的修改并不會對q造成影響。 結(jié)論: 編譯器總是要為函數(shù)的每個參數(shù)制作臨時副本,指針參數(shù)p的副本是 p,編譯器使 p = q(但是&p != &q,也就是他們并不在同一塊內(nèi)存地址,只是他們的內(nèi)容一樣,都是a的地址)。如果函數(shù)體內(nèi)的程序修改了p的內(nèi)容(比如在這里它指向b)。在本例中,p申請了新的內(nèi)存,只是把 p所指的內(nèi)存地址改變了(變成了b的地址,但是q指向的內(nèi)存地址沒有影響),所以在這里并不影響函數(shù)外的指針q。 這就需要二級指針操作: 2.二級指針操作 #include 這里只改了三個地方,變成傳二級指針。我們再看: 因為傳了指針q的地址(二級指針**p)到函數(shù),所以二級指針拷貝(拷貝的是p,一級指針中拷貝的是q所以才有問題),(拷貝了指針但是指針內(nèi)容也就是指針?biāo)赶虻牡刂肥遣蛔兊?/span>)所以它還是指向一級指針q(*p = q)。在這里無論拷貝多少次,它依然指向q,那么*p = &b;自然的就是 q = &b;了。3.再看一個例子: 我們代碼中以二級指針作為參數(shù)比較常見的是,定義了一個指針MyClass *ptr=NULL,在函數(shù)內(nèi)對指針賦值*ptr=malloc(...),函數(shù)結(jié)束后指針依然有效.這個時候就必須要用二級指針作為參數(shù)func(MyClass **p,...),一級指針為什么不行上面說了。 void my_malloc(char **s) { *s=(char*)malloc(100); } void main() { char *p=NULL; my_malloc(&p); //do something if(p) free(p); } 這里給指針p分配內(nèi)存,do something,然后free(p),如果用一級指針,那么就相當(dāng)于給一個p的拷貝s分配內(nèi)存,p依然沒分配內(nèi)存,用二級指針之后,才對p分配了內(nèi)存。 |
|