欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++浮點數(shù)在內(nèi)存中的存儲詳解

 更新時間:2022年01月12日 08:44:33   作者:七憶歲和  
大家好,本篇文章主要講的是C++浮點數(shù)在內(nèi)存中的存儲詳解,感興趣的同學趕快來看一看吧,對你有幫助的話記得收藏一下

前言:

我們在碼代碼的時候,經(jīng)常遇到過以整數(shù)形式存入,浮點數(shù)形式輸出;或者浮點數(shù)形式存入整數(shù)形式輸出。輸出的結果往往讓人意想不到,那么,為什么會發(fā)生這樣的變化,又是什么導致發(fā)生變化,接下來,就讓我們從存儲內(nèi)部結構出發(fā),帶你深度解刨!

我們以一個例子來說明一切

#include<stdio.h>
int main()
{
    int n = 9;
    float *pFloat = (float *)&n;
    printf("n的值為:%d\n",n);
    printf("*pFloat的值為:%f\n",*pFloat);
    *pFloat = 9.0;
    printf("n的值為:%d\n",n);
    printf("*pFloat的值為:%f\n",*pFloat);
    return 0;
}

這個例子我們將n以整數(shù)形式輸入,再定義一個指針變量來存n的地址,,分別將n以整數(shù)、浮點數(shù)形式輸出。

輸出結果如下

n和*pFloat在內(nèi)存中明明是同一個數(shù),為什么浮點數(shù)和整數(shù)的解讀結果會差別那么大?

要理解這個結果,一定要搞懂浮點數(shù)在計算機內(nèi)部的表示方法。

浮點數(shù)的表示形式

根據(jù)國際標準IEEE(電氣和電子工程協(xié)會)754,任意一個二進制浮點數(shù)V可以表示成下面的形式

(-1)^S*M*2^E(-1)^S表示符號位,當S=0,V為正數(shù);當S=1,V為負數(shù)。M表示有效數(shù)字,大于等于1,小于2.2^E表示指數(shù)位。

舉例來說:

十進制的5.0,寫成二進制是101.0,用科學計數(shù)法就是:1.01*2^2(類比于十進制的科學計數(shù)法:20000=2*10^4);

由于5.0是正數(shù),所以S=0,根據(jù)上面,M=1.01,E=2。

所以,寫成浮點數(shù)就是:(-1)^0*1.01*2^2。

浮點數(shù)存儲模型

IEEE754規(guī)定:

對于32位的浮點數(shù),最高的1位是符號位s,接著的8位是指數(shù)E,剩下的23位為有效數(shù)字M。

對于64位的浮點數(shù),最高的1位是符號位s,接著的11位是指數(shù)E,剩下的52位為有效數(shù)字M。

有效數(shù)字M

前面我們給了M的范圍是1≤M<2,那么為什么范圍是這么多?我們知道,二進制最高位為1,用科學計數(shù)法表示以后,M的形式為1.xxxxx的形式,其中xxxxx表示小數(shù)部分。

IEEE754規(guī)定,在計算機內(nèi)部保存M是,默認這個數(shù)的第一位總是1,因此可以被舍去,只保存后面的xxxxx部分。比如保存1.01的時候,只保存01,等到讀取的時候,再把第一位的1加上去。這樣做的目的是節(jié)省1位有效數(shù)字。以32位浮點數(shù)為例,留給M只有23位,將第一位的1舍去以后,等于可以保存24位有效數(shù)字。

指數(shù)E

至于指數(shù)E,情況就比較復雜。

首先,E為一個無符號整型(unsingde int)

這意味著,如果E為8位,它的取值范圍為0~255;如果E是11位,它的取值范圍為0~204。但是我們知道,科學計數(shù)法中的E是可以出現(xiàn)負數(shù)的,所以IEEE754規(guī)定,存入內(nèi)存時E的真實值必須再加上一個中間數(shù),對于8位的E,這個中間數(shù)是127;對于1位的E,這個中間數(shù)是1023。

比如:2^10的E是10,所以保存成32位浮點數(shù)時,必須保存成10+127=137,即化為二進制為10001001。

對于指數(shù)E從內(nèi)存中取出還可以再分成三種情況:

(1) E不全為0或不全為1

這時,浮點數(shù)就采用下面的規(guī)則表示,即指數(shù)E的計算值減去127(或1023),得到真實值,再將有效數(shù)字M前加上第一位的1。

比如:

0.5(1/2)的二進制形式為0.1,由于規(guī)定正數(shù)部分必須為1,即小數(shù)點右移1位,則為1.0*2^(-1),其階碼為-1+127=126,表示為01111110而尾數(shù)1.0去掉整數(shù)部分為0,補齊0到23位00000000000000000000000,則其二進制表示形式為:

0  01111110  00000000000000000000000

(2)E全為0

這時,浮點數(shù)的指數(shù)E等于1-127(或者1023)即為真實值,有效數(shù)字M不再加上第一位的1,而是還原為0.xxxxx的小數(shù)。這樣做是為了表示±0,以及接近于0的很小的數(shù)字。

(3)E全為1

這時,如果有效數(shù)字M全為0,表示±無窮大(正負取決于符號位S);

例題講解

(1)printf("n的值為:%d\n",n);

n是以整數(shù)存入的,在內(nèi)存中的存儲為:

輸出是整數(shù),所以 輸出的還是9。

(2)printf("*pFloat的值為:%f\n",*pFloat);

*pFloat相當于n的值,但輸出卻是以浮點數(shù)的形式輸出,但存入還是以整數(shù)形式存入,這是,內(nèi)存就會把存入的0 00000000 00000000000000000001001以浮點數(shù)形式處理,這里相當于,第一位就是S,即S=0,E全為0,剩下的為M,即M=00000000000000000001001。

(3)printf("n的值為:%d\n",n);

此時,*pFloat=9.0,也就是將9.0以浮點數(shù)形式存入,以整數(shù)形式取出。

根據(jù)前面,我們寫出9.0的存儲:

由于是正數(shù),所以S=0;

9轉換為二進制為1001,科學計數(shù)法:1.001*2^3;

∴E=3(屬于第一種,既不是全0,也不是全1),存入時要加上127,則E的真實值為:E=3+127=130,化為二進制1000 0010

M后面補0補全23位,即:001  0000 0000 0000 0000

所以浮點數(shù)在內(nèi)存中的存儲為:

 最后輸出以整數(shù)形式輸出,即把0 100000010 00100000000000000000000視為要輸出的數(shù)的補碼,但輸出的是原碼,因為最高位為0,所以這個數(shù)是正數(shù),原反補相同,則輸出這個數(shù),化為十進制就是:

 (4)printf("*pFloat的值為:%f\n",*pFloat);

以浮點數(shù)形式存入,以浮點數(shù)形式輸出,最后輸出還是9.000000,保留6位小數(shù)。

總結

到此這篇關于C++浮點數(shù)在內(nèi)存中的存儲詳解的文章就介紹到這了,更多相關C++ 浮點數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • C語言實現(xiàn)括號配對的方法示例

    C語言實現(xiàn)括號配對的方法示例

    本文主要介紹了C語言實現(xiàn)括號配對的方法示例,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-09-09
  • C語言中深度優(yōu)先搜索(DFS)算法的示例詳解

    C語言中深度優(yōu)先搜索(DFS)算法的示例詳解

    這篇文章主要通過兩個簡單的示例為大家詳細介紹一下C語言中深度優(yōu)先搜索(DFS)算法,文中的示例代碼講解詳細,感興趣的小伙伴可以了解一下
    2023-02-02
  • c/c++靜態(tài)庫之間相互調(diào)用的實戰(zhàn)案例

    c/c++靜態(tài)庫之間相互調(diào)用的實戰(zhàn)案例

    C++調(diào)用C的函數(shù)比較簡單,直接使用extern "C" {}告訴編譯器用C的規(guī)則去編譯C代碼就可以了,下面這篇文章主要給大家介紹了關于c/c++靜態(tài)庫之間相互調(diào)用的相關資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-08-08
  • C語言表達式求值中類型轉換和優(yōu)先級等問題詳解

    C語言表達式求值中類型轉換和優(yōu)先級等問題詳解

    表達式求值是一個常見的問題,可以用C語言實現(xiàn),下面這篇文章主要給大家介紹了關于C語言表達式求值中類型轉換和優(yōu)先級等問題的相關資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2023-05-05
  • 詳解C++11中的線程鎖和條件變量

    詳解C++11中的線程鎖和條件變量

    C++ 11允許開發(fā)者們以標準的、不依賴于平臺的方式編寫多線程程序。這篇文章概述了標準庫對于線程和同步操作機制的支持。這些都是非常重要的知識,希望讀者們可以認真看一下
    2021-06-06
  • C語言文件操作函數(shù)大全(超詳細)

    C語言文件操作函數(shù)大全(超詳細)

    本篇文章是對C語言中的文件操作函數(shù)進行了詳細的總結分析,需要的朋友參考下
    2013-05-05
  • C++設計類不能被繼承的方法實例講解

    C++設計類不能被繼承的方法實例講解

    在Java 中定義了關鍵字final,被final修飾的類不能被繼承,C++中如何實現(xiàn),下面我們來看一個例子
    2013-12-12
  • 從使用角度解讀c++20 協(xié)程示例

    從使用角度解讀c++20 協(xié)程示例

    類比線程,線程是個函數(shù),把這個函數(shù)交給 創(chuàng)建線程的api,然后這個函數(shù)就變成線程了,這個函數(shù)本身沒有任何特殊的地方,就是普通函數(shù),這篇文章主要介紹了從使用角度解讀c++20 協(xié)程示例,需要的朋友可以參考下
    2023-01-01
  • 詳解C語言初階基礎(2)

    詳解C語言初階基礎(2)

    這篇文章主要介紹了C語言中的初階基礎,介紹了其相關概念,具有一定參考價值。需要的朋友可以了解下,希望能夠給你帶來幫助
    2021-11-11
  • 詳解如何配置CLion作為Qt5開發(fā)環(huán)境的方法

    詳解如何配置CLion作為Qt5開發(fā)環(huán)境的方法

    這篇文章主要介紹了詳解如何配置CLion作為Qt5開發(fā)環(huán)境的方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-04-04

最新評論