c指標
初識c指標
1.什么是指標
2.指標和指標型別
3.野指標
4.指標運算
5.指標和陣列
6.二級指標
7.指標陣列
進階了解c指標
1.字符指標
2.陣列指標
3.指標陣列
4.陣列傳參和指標傳參
5.函式指標
6.函式指標陣列
7.指向函式指標陣列的指標
8.回呼函式
9.指標和陣列試題
開始之前,我們先來看一看這些東西
- int p; //這是一個整型變數
- int *pa; //首先從P 處開始,先與*結合,所以說明P 是一個指標,然后再與int 結合,說明指標所指向的內容的型別為int 型.所以Pa是一個回傳整型資料的指標
- int p[]; //首先從P 處開始,先與[]結合,說明P 是一個陣列,然后與int 結合,說明陣列里的元素是整型的,所以P 是一個由整型資料組成的陣列
- int *p[]; //首先從P 處開始,先與[]結合,因為其優先級比*高,所以P 是一個陣列,然后再與*結合,說明陣列里的元素是指標型別,然后再與int 結合,說明指標所指向的內容的型別是整型的,所以P 是一個由回傳整型資料的指標所組成的陣列
- int (*p)[]; //首先從P 處開始,先與*結合,說明P 是一個指標然后再與[]結合(與"()"這步可以忽略,只是為了改變優先級),說明指標所指向的內容是一個陣列,然后再與int 結合,說明陣列里的元素是整型的.所以P 是一個指向由整型資料組成的陣列的指標
進階
- int **p; //說明一下,這是個二級指標喔,首先從P 開始,先與*結合,說P 是一個指標,然后再與*結合,說明指標所指向的元素是指標,然后再與int 結合,說明該指標所指向的元素是整型資料.由于二級指標以及更高級的指標極少用在復雜的型別中,所以后面更復雜的型別我們就不考慮多級指標了,最多只考慮一級指標.
- int p(int); //從P 處起,先與()結合,說明P 是一個函式,然后進入()里分析,說明該函式有一個整型變數的引數,然后再與外面的int 結合,說明函式的回傳值是一個整型資料
- Int (*p)(int); //從P 處開始,先與指標結合,說明P 是一個指標,然后與()結合,說明指標指向的是一個函式,然后再與()里的int 結合,說明函式有一個int 型的引數,再與最外層的int 結合,說明函式的回傳型別是整型,所以P 是一個指向有一個整型引數且回傳型別為整型的函式的指標
- int *(*p(int))[3]; //可以先跳過,不看這個型別,過于復雜從P 開始,先與()結合,說明P 是一個函式,然后進入()里面,與int 結合,說明函式有一個整型變數引數,然后再與外面的*結合,說明函式回傳的是一個指標,,然后到最外面一層,先與[]結合,說明回傳的指標指向的是一個陣列,然后再與*結合,說明陣列里的元素是指標,然后再與int 結合,說明指標指向的內容是整型資料.所以P 是一個引數為一個整資料且回傳一個指向由整型指標變陣列成的陣列的指標變數的函式.
看了看上面的東西,是不是感覺指標并沒有那么難,減少對指標的恐懼,好啦,現在開始我們的主題,
第一部分
初識c指標
1.什么是指標?
在計算機科學中,指標是編程語言中的一個物件,利用地址,它的值直接指向存在電腦存盤器中另一個地方的值,由于通過地址能找到所需的變數單元,可以說,地址指向該變數單元,因此,將地址形象化稱為:指標,意思為它能找到以它為地址的記憶體單元
我們可以這樣理解:記憶體是一塊大的空間,這么大的空間我們該怎么去管理?把記憶體劃分為一個個記憶體單元,取一個位元組為基礎的記憶體單元,為了很好管理,我們把這些記憶體進行編號,這個編號在記憶體中也叫地址,把地址也形象稱為指標,就這樣記憶體編號,記憶體地址,指標實際上都是一回事,是不是概念更加清楚了,
實際動手一下,比如int a = 10;//占4個位元組空間,&a拿到的是a的4個位元組中第一個位元組的地址,int*pa = &a;*說明pa是指標變數,然后int說明pa指向的物件是整型型別*pa = 20;通過pa就能很好的找到a,這時候20就能改掉,這是上面第2點講到的,地址存起來需要一個變數,既存放指標,所有把存放地址的變數稱為指標變數,
總結一下:指標就是變數,用來存放地址,指標是多大的呢?指標的大小在32位平臺是4個位元組,在64位平臺是8個位元組,不知道大家知不知道這個點,這里稍微提一下,
2.指標和指標型別
?????為什么講這個呢,下面我們來看一下代碼和運行結果
大小都是4個位元組啊,那跟型別有什么關系,型別好像沒什么用,型別沒什么用,我們造一個通用型別不就行了?用一個代表所有不是很簡單??此時我們心中一萬個疑問涌上,實際上并沒有存在,這反過來恰恰說明了一個點,這些型別是有用的,不會平白無故的出現,意義是什么?
這里很有必要說一說指標型別的意義,我看過別人寫過指標的博客,但是好像都沒有說到這里
指標型別的意義
1.指標型別決定了解參考的權限有多大 ,int解參考能訪問4個位元組 char解參考能訪問1個位元組
2.指標型別決定了,指標走一步,能走多遠(步長),這這里給個代碼大家好好看一下

總結:指標的型別決定了指標向前或者向后走一步有多大以及權限的大小,這就是指標型別的兩個意義,不是胡亂的出現的,這一點可以提高大家的理解,
3.野指標
野指標就是指標指向的位置是不可知的(隨機的,不正確的,沒有明確限制的)
原因
1.指標未初始化

2.指標越界訪問

3.指標指向的空間釋放

回傳a后生命周期結束,銷毀還給作業系統,*p訪問空間的時候出問題了,
如何規避野指標?
指標初始化
吐槽一下,課本上很多東西都沒初始化,這些需要我們注意,初始化是良好的編程風格
小心指標越界
指標指向空間釋放既使置NULL
使用前檢查有效性
4.指標運算
指標+-整數
指標-指標
指標的關系運算
指標的關系運算是比較大小

這段代碼用指標+整數以及指標的關系運算,還有一個指標-指標

實際上,指標-指標得到的兩個指標之間的元素個數,指標和指標相減的前提是:兩個指標指向同一塊空間
這里提一個標準規定:
允許指向陣列元素的指標指向陣列最后一個元素后面的那個記憶體位置的指標比較,但是不允許與指向第一個元素之前的那個記憶體位置的指標進行比較,
5.指標和陣列
陣列名是什么?
陣列名是首元素的地址,這是我們反復強調的,這里可以理解一下,問題不大


從這個代碼我想告訴大家一個點,首元素地址+i產生的就是下標為i的地址 ,
在來看一個有趣的

//arr[2]----*(arr+2)----*(2+arr)---2[arr]
到這里大家有沒有對陣列有了更深層次的理解呢
6.二級指標
有了這樣的理解之后,我們在來看看二級指標
什么是二級指標呢?
二級指標
來看一小段代碼

ppa首先是指標既*ppa,ppa指向的物件是pa,pa的型別整體是int*,所以就是int* *ppa,書寫上有區別,但是實際并沒有區別,ppa這就是二級指標變數,希望大家能夠理解, 實際上就是套娃....
就是說*ppa == pa | *pa == a | **ppa == a,兩層關系,這就是二級指標
看到這里,我們很容易聯想到三級指標,繼續套娃

當然,你也可以搞出四級指標,五級指標,語法是支持的,但是很少用到,沒必要深究
看到這里,你是不是對二級指標有了更深的理解,
最后,我們在來學習一個知識點
7.指標陣列
指標陣列是什么東西呢?
指標陣列實際上----陣列
怎么說呢?
先來了解一下,整型陣列——存放整型的陣列就是整型陣列
字符陣列——存放的是字符
以此類推
指標陣列——存放指標的陣列
前面剛開始是第4點有提到,是不是就串起來了呢,當然,我們可以看看這個代碼
parr[5]是一個陣列,存放整型指標的陣列,大家應該理解了把
好啦,到這里就介紹完初始c指標的基礎點,此時你已經有了對付c指標的初步知識,有了初步了解,接下去,進入我們第二部分,進階版本,
進階了解c指標
首先恭喜你成功看到了這里,實際上,第一部分比較簡單,學的很淺接下來,介紹第二部分進階了解c指針,帶大家進一步去了解,希望大家能跟上節奏,這個部分我們來說說指標的高級主題
1.字符指標
在指標的型別中我們知道有一種指標型別為字符指標char*;這是很普遍的,很多人不理解為什么要放在這里,不急,接下去看,
之前我們都是這樣子用字符指標的

但是我想說的是,字符指標不僅僅可以指向一個字符,還可以指向一個字串

其實,這種寫法是把字串首字符h地址放在ps上,可以看我們的運行結果
在來看一看這個代碼,是不是頭腦更加清晰了,慢慢感受把,是不是對字符指標有了更深的認知,

有必要提醒一下,常量字串是不能去改的,注意const的修飾,使語法更加標準一些
指標陣列
上面也是已經提過,在來說一說,下面這種寫法比較奇怪

在來看看這個,模擬二維陣列的實作
也可以寫成這樣子

這就是指標陣列的用處
陣列指標
陣列指標的定義?
看到這里是不是懵了,指標陣列??陣列指標??
陣列指標是指標??還是陣列??心里頭一萬只草泥馬走過
答案是:指標
陣列指標是一種指標,指向陣列的指標,下面給大家好好推敲推敲,不要著急,慢慢來

parr存放的就是陣列的地址,這就是陣列指標

p1是整型指標,p2是陣列指標,剛開始都是起始地址,結果相同,看看運行結果可知,都加1結果肯定不同,一個是整型指標,另一個是陣列指標,結果肯定不同啊,可以看看運行結果
注意:&陣列名取出的是整個陣列的地址,不是首元素的地址!!
到了這里,有必要總結一下
陣列名是首元素地址
但是有2個例外
1.sizeof(陣列名)-陣列名表示整個陣列,計算的是整個陣列大小,單位是位元組
2.&陣列名--陣列名表示整個陣列,取出的是整個陣列的地址
希望大家牢記起來,不要老是一度認為陣列名就是首元素地址,到時候錯了都不知道怎么錯的,
學了陣列指標和指標陣列我們來看看下面代碼的意思
int arr[5];——整型陣列
int*parr1[10]——整型指標陣列
int(*parr2)[10]——陣列指標,該指標能夠指向一個陣列,陣列10個元素,每個元素型別都是int
int(*parr3[10])[5]——parr3是一個存盤陣列指標的陣列,該陣列能夠存放10個陣列指標,每個陣列指標能夠指向一個陣列,陣列5個元素,每個元素是int型別
陣列傳參和指標傳參
寫代碼難免把陣列和指標傳給函式,函式該怎么設計呢
一維陣列傳參
二維陣列傳參
一級指標傳參
二級指標傳參
這些都要我們去注意了解,怎么去接收?這里就不展開討論了,這里只是適當的給大家拓展一下
遇到的時候多加留意,不懂的也可以看看書籍相關知識,不過這個比較少用到,這里跟大家說說有這回事,
到了這里給大家梳理一下,以免大家開始亂了,實際上,理清思路有助于我們更好的記憶:
指標
一級指標,二級指標,陣列指標
陣列
一維陣列,二維陣列,指標陣列
接下去,進入我們的函式指標
函式指標
指向函式的指標就叫函式指標,這些概念其實本質上是一樣的

這便是函式指標,從上面代碼我們還可以看到 函式名==&函式名,這是完全等價的,但是
陣列名!=&陣列名

pf就是函式指標變數,跟陣列指標的表示方法非常類似
這里有必要加固一下表示的練習,如圖所示

怎么呼叫呢,我們來實際看看

我們在來看一個東西,實際上Add == pf

這個結果也是8,
函式指標陣列
怎么去理解函式指標陣列呢,直接擺上我們的代碼,里面有注釋了,大家耐心觀看,便會懂了,實際并沒有那么玄妙
pfArr就是所謂的函式指標陣列,當然,Add與Sub可以用pf1與pf2代替,到這里,你對它已經有了認識,認識到了概念,但是怎么用呢?有什么用,這里,非要講用處的話,我可以舉例子給大家看看,函式指標陣列的運用,這里以簡單的計算器加減乘除為例,為大家簡單運用一下函式指標陣列

這里大大簡化了程式,你可以試試用switch陳述句去改寫,你就知道這個變得多簡潔了,
接下來,再給大家說一個東西
指向函式指標陣列的指標
????剛剛說的是函式指標的陣列--陣列
取出函式指標陣列的地址
&函式指標陣列
int(*p)(int,int);//函式指標
int(*p2[4])(int,int);//函式指標陣列
p3 = &p2;//取出的是函式指標陣列的地址
//p3就是一個指向函式指標的陣列的指標
那么問題來了,p3怎么去定義呢
int(*(*p3)[4])(int,int) = &p2;
大家了解一下就ok了,不要重點研究,實際上指標就是在套娃,一直在套娃,這東西已經比較深入了,學不會也沒關系,
回呼函式
回呼函式這塊我本來不想說的,一方面是這篇博客寫的時間已經用了很久,另一方面感覺沒必要,沒啥精力寫了,但是既然到了這里,我還是介紹一下把,回呼函式就是一個通過函式指標呼叫的函式,

通過函式指標呼叫,這里只是簡單運用一下,
冒泡排序c語言實作_平凡的人1的博客-CSDN博客這里大家可以看看我之前寫過的博客
我這里主要是想說一個函式qsort()//快速排序,實際上,關于用到函式自己卻不懂的時候
我給大家推薦一款東西MSDN,查找相關函式怎么用,有什么引數,或者去一個網站,cplusplus.com,大家可以去看看

指標和陣列試題



好啦,手工結束
轉載請註明出處,本文鏈接:https://www.uj5u.com/qita/354638.html
標籤:其他
