1、先把問題擺出來
參考Demo:
1#include
運行結果:
現象描述:
大家可以看到當試圖輸出**((int **)ptr)的時候程序奔潰了,其實在bug菌剛開始學習C的時候也是遇到了這個問題,最終理解清楚了就自然明白了。粗暴點就把((int **)ptr);*((int **)ptr);**((int **)ptr)都嘗試著打印出來分析分析。
所以bug菌在下面把指針的一些知識點都跟大家講解一下,形成系統的知識,避免一些小伙伴僅學習了一些碎片而一知半解。
2、二級指針的使用
1
多級指針 其實多級指針在嵌入式程序中的應用還是相對比較少的,超過3級的bug菌僅僅只在一個特殊的索引功能里面使用過。 這里重點看看一、二級,畢竟二級指針與我們的二維數據結合使用,二維素組在圖形、矩陣、算法等等方面還是使用非常廣泛的。
2
一級指針 指針其實廣義的講它也是一種數據類型,所以所謂的int* ptr;其中int* 就是變量ptr的類型,那么ptr就是我們常說的指針變量。
Demo:
1#include
運行結果:
圖解:
分析一下:
ptr既然是變量,變量存于內存中,那么就一定有其地址,如上圖所示ptr位于0x28FEE8地址處,其中其ptr里面保存的就是藍色區域中的地址,也就是a變量所在的地址,所以&a與ptr是相等的。
*ptr就很好理解了,你可以把*ptr看成一個變量,其類型為int,其變量位于ptr值所在的內存地址處,即0x28FEEC處的int變量與定義的int a剛好一致。
3
二級指針 如果大家已經理解了一級指針,二級指針也就順理成章了,來看看下面的小程序:
Demo:
1#include
運行結果:
圖解:
分析一下:
通過上圖來看ptrptr也是一個變量,其類型為int**,變量肯定有內存,其地址就是0x28FEE4,其變量保存的值是0x28FEE8(即ptr的地址)。
那么*ptrptr,同樣跟一級指針一致,把*ptrptr看成一個變量,也就是在ptrptr的值0x28FEEC8地址處的一個int*類型的變量,且該變量的值是0x28FEEC,其自身的地址為0x28FEE8。
同樣對于**ptrptr也看成變量,也就是*ptrptr的值0x28FEEC地址處的一個int類型的變量,那么此時該變量與a是相等的。
最后在解釋一下&*ptrptr,由于*ptrptr一個變量,&*ptrptr表示該變量的地址即0x28FEE8,然而該值剛好也是ptrptr變量的值,所以&*ptrptr =ptrptr。
3、是時候解答前面的問題了
參考Demo:
1#include
分析一下:
我們知道問題出在**(int**)ptr,不太理解的小伙伴總是覺得,我已經強制類型轉化為二級指針了,前面只需要用**獲得最后的值即可,怎么就不行呢? 我知道這里有問題我就是不理解為什么?
那我們一起通過前面的知識來分析分析。同樣把ptr看成是變量,變量的值并不會跟隨強制類型而發生改變。強制類型僅僅只是改變了獲取內存中數據的方式,并沒有改變內存中的數據。下面圖解一下:
所以本例子中對于強制類型轉化中的**(int*****)ptr,無論強制類型轉化為幾級指針都沒有絲毫意義,因為變量本身的值沒有發生變化!所以強制類型轉化以后第二個*便會指向出問題,從而導致訪問了不正確的內存空間而程序奔潰。
多級指針強制類型轉化的目的大部分都是為了滿足編譯器檢查指針層級操作邏輯是否有誤。
以后大家對于多級指針的分析不太熟練的話可以跟bug菌一樣畫畫圖分析分析,基本上一些理解上的問題就迎刃而解了。
4、最后小結
指針的理解就為大家講解到這里了,如果你對指針還有畏懼感,那就只有一個可能,使用得太少了,多加練習自然生巧!
-
算法
+關注
關注
23文章
4630瀏覽量
93355 -
矩陣
+關注
關注
0文章
425瀏覽量
34643 -
嵌入式程序
+關注
關注
0文章
6瀏覽量
6566
原文標題:C進階技巧:二級指針這個問題被問好幾遍!該終結了!
文章出處:【微信號:gh_c472c2199c88,微信公眾號:嵌入式微處理器】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
評論