深入理解C語言指針及占據(jù)內(nèi)存空間
原文鏈接:https://www.cnblogs.com/l-hh/p/12288613.html
第一、了解內(nèi)存空間
本文章文字有點多,會有點枯燥,配合圖文一起看可以緩解枯燥,耐心閱讀哦?。?!
先了解內(nèi)存地址,才更好的理解指針!
我們可以把內(nèi)存想象為成一列很長很長的貨運火車,有很多大小相同的車廂,而每個車廂正好相當(dāng)于在內(nèi)存中表示一個字節(jié)
。這些車廂裝著不同的貨物,就像我們的內(nèi)存要存著各式各樣的數(shù)據(jù)。
多啰嗦一下
我們平時在電腦上能夠聽音樂、看視頻和文章,其實看到的這些東西就是內(nèi)存中每個“車廂”里面的數(shù)據(jù),這些數(shù)據(jù)最終還是由二進制0/1演變而成。
雖然視頻、文章、音樂等這些信息在我們眼里是不同的,但對于計算機來說它們在內(nèi)存中都是以二進制的形式來表示。
因為我們要知道去哪存或取數(shù)據(jù),所以內(nèi)存中每個字節(jié)都有對應(yīng)的編號,就像火車上的車廂編號一樣。而這個內(nèi)存中每個字節(jié)的編號就是我們常說的內(nèi)存地址,是按一個字節(jié)接著一個字節(jié)的次序進行編址。如下圖所示:
凡事多問幾個為什么?
1. 為什么內(nèi)存地址都有0x開頭?
0x 開頭代表以十六進制來表示的意思。
2. 為什么我們平時看到內(nèi)存地址是這樣的呢?如圖:
因為內(nèi)存容量很大,容量大字節(jié)數(shù)自然也多了,所以需要更多位來編址內(nèi)存地址。上圖的(0x00 ...)內(nèi)存地址這里只是便于理解!
3. 為什么我那么菜呢?
哈哈哈......你心里沒點*數(shù)嗎?
關(guān)于內(nèi)存字節(jié)
- 1個內(nèi)存地址只存1個字節(jié) (Byte);
- 1個字節(jié)等于8位二進制,每一位二進制的0或1,叫“比特”(bit);
- 比特是最小單位,字節(jié)是比特的集合,也是一個單位;
內(nèi)存給數(shù)據(jù)類型地址分配如下:
- char:占一個字節(jié)分配一個地址;
- int: 占四個字節(jié)分配四個地址;
- 還有l(wèi)ong、float、double等類型,等著你來動手測試。
可以使用sizeof進行驗證:
#include<stdio.h> int main () { printf("sizeof(char)=%u\n",sizeof(char)); printf("sizeof(int)=%u\n",sizeof(int)); return 0; }
結(jié)果如下:
第二、理解指針
不要把指針想得太復(fù)雜,指針的實質(zhì)就是內(nèi)存“地址”,可以說指針就是地址,其實指針就是保存地址的變量。
拿普通變量跟指針變量做比較:
char a; // 定義一個變量a,用于保存char類型的數(shù)據(jù); char *b; // 定義一個指針變量b,用于保存一個內(nèi)存地址,這個內(nèi)存地址上的數(shù)據(jù)必須是char類型的。
舉個例子,給指針變量進行賦值:
#include<stdio.h> int main () { char a = 5; // char 類型占一個字節(jié); char *b = &a; // “&”是取變量的地址,取出a在內(nèi)存中的地址; // 賦值給b指針,此時b變量存儲的就是a地址。 printf("我是a變量的值:%d\n",*b); // *b表示輸出b里面存儲的地址上的數(shù)據(jù); // 證明b上存儲的是a的地址; printf("我是a的地址:%p\n",&a); printf("我是b變量的值:%p\n",b); return 0; }
輸出結(jié)果為:
我是a變量的值:5 我是a的地址:000000000062FE17 我是b變量的值:000000000062FE17
通過畫圖來理解:
通過指針間接性修改變量的值
char a = 5; char *b = &a; printf("初始值:a=%d,*b=%d\n",a,*b); *b = 12; // 其實操作的就是變量a本身的值; printf("修改后:a=%d,*b=%d\n",a,*b); ------------------------------------------ 輸出結(jié)果為: 初始值:a=5,*b=5 修改后:a=12,*b=12
指針類型的概念
我們知道char類型的數(shù)據(jù)只占一個字節(jié),有很多類型是需要多個字節(jié)來存儲的,像int類型的數(shù)據(jù)就需要四個字節(jié)來存儲(根據(jù)平臺不同,長度也有可能不一致)。
對于int類型的指針從當(dāng)前字節(jié)(地址)開始共四個字節(jié)(地址)都是屬于該變量的值, 而對于char類型則只表示當(dāng)前字節(jié)(地址)。代碼如下:
int a = 259; int * p1 = &a; char * p2 = (char *)&a; // 這里需要強制轉(zhuǎn)換一下類型 printf("*p1=%d,*p2=%d\n",*p1,*p2); ----------------------- 輸出:*p1=259,*p2=3
通過畫圖來便于理解:
通過上文我們已經(jīng)對int類型指針有所了解了,*p1的輸出是在我們預(yù)算范圍之內(nèi)的,但是為什么*p2輸出的值是3呢?
重點,敲黑板?。?!
因為計算機是使用二進制來表示數(shù)字的,上面(259)十進制轉(zhuǎn)換二進制是 [100000011],由于一個int類型變量占用四個字節(jié),8位二進制為一個字節(jié),補齊高位的0后,則 [00000000 00000000 00000001 00000011],每8位二進制(一個字節(jié))換算為十進制,則 [0 0 1 3]。
相關(guān)文章
C++?ffmpeg實現(xiàn)將視頻幀轉(zhuǎn)換成jpg或png等圖片
有時播放實時流的時候有截圖的需求,需要將解碼出來的圖片保存本地或上傳服務(wù)器,這時就需要將avframe中的數(shù)據(jù)編碼成png、jpg等格式的圖片,我們使用ffmpeg的相關(guān)編碼器就可以實現(xiàn)功能,下面就來講講具體實現(xiàn)方法吧2023-03-03