C語(yǔ)言數(shù)據(jù)的存儲(chǔ)超詳細(xì)講解下篇浮點(diǎn)型在內(nèi)存中的存取
前言
本文接著學(xué)習(xí)數(shù)據(jù)的存儲(chǔ)相關(guān)的內(nèi)容,主要學(xué)習(xí)浮點(diǎn)型數(shù)在內(nèi)存中的存儲(chǔ)與取出。
浮點(diǎn)型在內(nèi)存中的存儲(chǔ)
- 常見(jiàn)的浮點(diǎn)數(shù):3.14159、1E10
- 浮點(diǎn)數(shù)家族包括: float、double、long double 類型
- 浮點(diǎn)數(shù)表示的范圍:float.h中定義
浮點(diǎn)數(shù)存儲(chǔ)的例子
int main() { int n = 9; float *pFloat = (float*)&n; printf("n的值為:%d\n",n); printf("*pFloat的值為:%f\n",*pFloat); *pFloat = 9.0; printf("num的值為:%d\n",n); printf("*pFloat的值為:%f\n",*pFloat); return 0; }
乍得一看,輸出結(jié)果是: 9 、9.0、9、9.0
運(yùn)行結(jié)果見(jiàn)下圖:
浮點(diǎn)數(shù)存儲(chǔ)規(guī)則
num 和 *pFloat 在內(nèi)存中明明是同一個(gè)數(shù),但是浮點(diǎn)數(shù)和整數(shù)的解讀結(jié)果會(huì)差別巨大,要理解這個(gè)結(jié)果,需要理解浮點(diǎn)數(shù)在計(jì)算機(jī)內(nèi)部的表示方法。
根據(jù)國(guó)際標(biāo)準(zhǔn)IEEE(電氣和電子工程協(xié)會(huì)) 754,任意一個(gè)二進(jìn)制浮點(diǎn)數(shù)V可以表示成下面的形式:
- (-1)^S * M * 2^E
- (-1)^s表示符號(hào)位,當(dāng)s=0,V為正數(shù);當(dāng)s=1,V為負(fù)數(shù)
- M表示有效數(shù)字,大于等于1,小于2
- 2^E表示指數(shù)位
舉例說(shuō)明:
- 十進(jìn)制的5.0,寫成二進(jìn)制是 101.0 ,相當(dāng)于 1.01×2^2
- 按照上面的存儲(chǔ)規(guī)則,可以得出s=0,M=1.01,E=2。
- 十進(jìn)制的-5.0,寫成二進(jìn)制是 -101.0 ,相當(dāng)于 -1.01×2^2
- 按照上面的存儲(chǔ)規(guī)則,s=1,M=1.01,E=2
IEEE 754規(guī)定
對(duì)于32位的浮點(diǎn)數(shù),最高的1位是符號(hào)位s,接著的8位是指數(shù)E,剩下的23位為有效數(shù)字M
對(duì)于64位的浮點(diǎn)數(shù),最高的1位是符號(hào)位S,接著的11位是指數(shù)E,剩下的52位為有效數(shù)字M
IEEE 754對(duì)有效數(shù)字M的特別規(guī)定
- 規(guī)定中1≤M<2 ,也就是說(shuō),M可以寫成 1.xxxxxx 的形式,其中xxxxxx表示小數(shù)部分
- 在計(jì)算機(jī)內(nèi)部保存M時(shí),默認(rèn)這個(gè)數(shù)的第一位總是1,因此可以被舍去,只保存后面的xxxxxx部分
- 例如保存1.01的時(shí)候,只保存01,等到讀取的時(shí)候,再把第一位的1加上去。這樣做的目的,是節(jié)省1位有效數(shù)字
- 以32位浮點(diǎn)數(shù)為例,留給M只有23位,將第一位的1舍去以后,等于可以保存24位有效數(shù)字
IEEE 754對(duì)指數(shù)E的特別規(guī)定
存入內(nèi)存是E的規(guī)定
E為一個(gè)無(wú)符號(hào)整數(shù)(unsigned int)
- 如果E為8位,它的取值范圍為0-255
- 如果E為11位,它的取值范圍為0~2047
但是,科學(xué)計(jì)數(shù)法中的E是可以出現(xiàn)負(fù)數(shù)的,所以IEEE 754規(guī)定,存入內(nèi)存時(shí)E的真實(shí)值必須再加上一個(gè)中間數(shù):
- 對(duì)于8位的E,這個(gè)中間數(shù)是127
- 對(duì)于11位的E,這個(gè)中間數(shù)是1023
- 例如,2^10的E是10,所以保存成32位浮點(diǎn)數(shù)時(shí),必須保存成10+127=137,即10001001
從內(nèi)存取出時(shí)E的規(guī)定
1、E不全為0或不全為1
- 浮點(diǎn)數(shù)就采用下面的規(guī)則表示,即指數(shù)E的計(jì)算值減去127(或1023),得到真實(shí)值,再將有效數(shù)字M前加上第一位的1
- 例如:0.5(1/2)的二進(jìn)制形式為0.1,由于規(guī)定正數(shù)部分必須為1,即將小數(shù)點(diǎn)右移1位,則為1.0*2^(-1),其階碼為-1+127=126,表示為01111110
- 尾數(shù)1.0去掉整數(shù)部分為0,補(bǔ)齊0到23位00000000000000000000000,則其二進(jìn)制表示形式為0 01111110 00000000000000000000000
2、E全為0
- 浮點(diǎn)數(shù)的指數(shù)E等于1-127(或者1-1023)即為真實(shí)值
- 有效數(shù)字M不再加上第一位的1,而是還原為0.xxxxxx的小數(shù)。這樣做是為了表示±0,以及接近于0的很小的數(shù)字
3、E全為1
如果有效數(shù)字M全為0,表示±無(wú)窮大(正負(fù)取決于符號(hào)位s)
舉例 1
int main() { float f = 5.5; //浮點(diǎn)數(shù) 101.1 二進(jìn)制表示 (-1)^0 * 1.011* 2^2 IEEE 745規(guī)定 s=0 //代表正數(shù) E=2 //代表指數(shù),左移2位 ,存儲(chǔ)是時(shí)要+127 =129 M=1.011 //有效數(shù)字 0 10000001 01100000000000000000000 0100 0000 1011 0000 0000 0000 0000 0000 0x40 b0 00 00 小端存儲(chǔ) }
由上面分析可知,浮點(diǎn)數(shù)5.5在內(nèi)存的存儲(chǔ)形式見(jiàn)下圖:
調(diào)試程序發(fā)現(xiàn)結(jié)果與分析過(guò)程一致,并且是小端存儲(chǔ)。
舉例 2
int main() { float f = 0.5; 浮點(diǎn)數(shù) 0.1 二進(jìn)制表示 (-1)^0 * 1.0*2^-1 IEEE 745規(guī)定 S = 0 代表正數(shù) M = 1.0 有效數(shù)字 E = -1 代表指數(shù),右移1位 ,存儲(chǔ)是要+127 =126 0 01111110 00000000000000000000000 0011 1111 0000 0000 0000 0000 0000 0000 0x3f 00 00 00 return 0; }
由上面分析可知,浮點(diǎn)數(shù)0.5在內(nèi)存的存儲(chǔ)形式見(jiàn)下圖:
調(diào)試程序發(fā)現(xiàn)結(jié)果與分析過(guò)程一致,并且是小端存儲(chǔ)。
舉例 3
下面對(duì) 3.1的例子進(jìn)行講解:
int main()
{
int n = 9;
第一步:正數(shù)9在內(nèi)存存儲(chǔ)的形式:
00000000000000000000000000001001
float *pFloat = (float*)&n;
第二步:將正數(shù)強(qiáng)制轉(zhuǎn)換位浮點(diǎn)型,認(rèn)為pfloat指向的內(nèi)容是浮點(diǎn)數(shù)
存儲(chǔ)在內(nèi)存中的形式
0 00000000 00000000000000000001001
0000 0000 0000 0000 0000 1001
0x00 00 00 09
s=0
E= -126 因?yàn)镋是全為0的特殊情況,取出就是1-127固定的
M= 0.00000000000000000001001 后面23位都是小數(shù)位
第三步:從內(nèi)存中取出浮點(diǎn)數(shù)
(-1)^0 * 0.00000000000000000001001 * 2^-126
結(jié)果為極限接近0的非常小的數(shù)
printf("n的值為:%d\n",n); 輸出9
printf("*pFloat的值為:%f\n",*pFloat);輸出浮點(diǎn)數(shù)0.00000
*pFloat = 9.0;
第一步:浮點(diǎn)數(shù)9.0的二進(jìn)制形式:
1001.0
(-1)^0 * 1.001 * 2^3
s=0
E=3 代表指數(shù),左移3位 ,存儲(chǔ)是要+127 =130
M=1.001
第二步:浮點(diǎn)數(shù)9.0在內(nèi)存存儲(chǔ)的形式:
0 10000010 00100000000000000000000
0100 0001 0001 0000 0000 0000 0000
0x41 10 00 00
第三步:%d打印,上面的補(bǔ)碼就是正數(shù)的補(bǔ)碼了,三碼合一
打印原碼: 1,091,567,616
printf("num的值為:%d\n",n);
printf("*pFloat的值為:%f\n",*pFloat); 9.0
return 0;
}
此時(shí)再看結(jié)果,就會(huì)一目了然了:
- 輸出浮點(diǎn)數(shù)0.00000,浮點(diǎn)數(shù)在內(nèi)存的存儲(chǔ)形式 0x00 00 00 09,小端存儲(chǔ)形式:
輸出正數(shù)1,091,567,616,正數(shù)數(shù)在內(nèi)存的存儲(chǔ)形式 0x41 10 00 00,小端存儲(chǔ)形式:
輸出結(jié)果與分析一致:
判斷兩個(gè)浮點(diǎn)數(shù)是否相等?
兩個(gè)浮點(diǎn)數(shù)不能直接判斷是否相等,應(yīng)該判斷他們之間的差值是否在一個(gè)給定范圍內(nèi),滿足自己的使用要求即可。
int main() { int a = 0; if (a == 1)//整數(shù)可以直接判斷 { } float b = 0.00001;//基本接近0,但不是0 if (b==0.0)//不能這樣判斷,會(huì)出問(wèn)題 { } }
總結(jié)
數(shù)據(jù)的存儲(chǔ)相關(guān)內(nèi)容是C語(yǔ)言進(jìn)階階段的第一個(gè)知識(shí)點(diǎn),與整形提升關(guān)系密切,還要熟悉變量類型、符號(hào)位、類型范圍、原碼反碼補(bǔ)碼、等等,這部分內(nèi)容更多的牢記變量存儲(chǔ)類型的性質(zhì),不能想當(dāng)然的去考慮結(jié)果,每一步的思考都要有依據(jù)才行。
要溫故而知新,通過(guò)這部分的學(xué)習(xí),給自己在以后的程序找錯(cuò)中,提供了不一樣的思路。
下一篇開(kāi)始學(xué)習(xí)指針進(jìn)階的內(nèi)容了。(鏈接直達(dá))
到此這篇關(guān)于C語(yǔ)言數(shù)據(jù)的存儲(chǔ)超詳細(xì)講解下篇浮點(diǎn)型在內(nèi)存中的存取的文章就介紹到這了,更多相關(guān)C語(yǔ)言 浮點(diǎn)型數(shù)據(jù)的存儲(chǔ) 內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++順序容器(vector、deque、list)的使用詳解
本文主要介紹了C++順序容器(vector、deque、list)的使用詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06C++實(shí)現(xiàn)停車場(chǎng)管理系統(tǒng)的示例代碼
停車場(chǎng)管理系統(tǒng)就是模擬停車場(chǎng)進(jìn)行車輛管理的系統(tǒng),該系統(tǒng)分為汽車信息模塊,用戶使用模塊和管理員用戶模塊,本文將用C++實(shí)現(xiàn)這一簡(jiǎn)單的系統(tǒng),希望對(duì)大家有所幫助2023-04-04C語(yǔ)言實(shí)現(xiàn)飛機(jī)售票系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)飛機(jī)售票系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-05-05簡(jiǎn)述C語(yǔ)言中system()函數(shù)與vfork()函數(shù)的使用方法
這篇文章主要介紹了簡(jiǎn)述C語(yǔ)言中system()函數(shù)與vfork()函數(shù)的使用方法,是C語(yǔ)言入門學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下2015-08-08C語(yǔ)言中qsort函數(shù)用法實(shí)例小結(jié)
這篇文章主要介紹了C語(yǔ)言中qsort函數(shù)用法,包括了針對(duì)各種數(shù)據(jù)類型參數(shù)的排序,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2014-09-09