老大說:誰要再用double定義商品金額宣肚,就自己收拾東西走

image

先看現(xiàn)象

涉及諸如float或者double這兩種浮點(diǎn)型數(shù)據(jù)的處理時(shí)导坟,偶爾總會(huì)有一些怪怪的現(xiàn)象屿良,不知道大家注意過沒,舉幾個(gè)常見的栗子:

典型現(xiàn)象(一):條件判斷超預(yù)期

System.out.println( 1f == 0.9999999f );   // 打颖怪堋:false
System.out.println( 1f == 0.99999999f );  // 打映揪濉:true    納尼?

典型現(xiàn)象(二):數(shù)據(jù)轉(zhuǎn)換超預(yù)期

float f = 1.1f;
double d = (double) f;
System.out.println(f);  // 打拥莸荨:1.1
System.out.println(d);  // 打优绯取:1.100000023841858  納尼?

典型現(xiàn)象(三):基本運(yùn)算超預(yù)期

System.out.println( 0.2 + 0.7 );  

// 打拥俏琛:0.8999999999999999   納尼贰逾?

典型現(xiàn)象(四):數(shù)據(jù)自增超預(yù)期

float f1 = 8455263f;
for (int i = 0; i < 10; i++) {
    System.out.println(f1);
    f1++;
}
// 打印:8455263.0
// 打硬っ搿:8455264.0
// 打痈斫!:8455265.0
// 打印:8455266.0
// 打蛹:8455267.0
// 打友早汀:8455268.0
// 打印:8455269.0
// 打咏啤:8455270.0
// 打庸苄:8455271.0
// 打印:8455272.0

float f2 = 84552631f;
for (int i = 0; i < 10; i++) {
    System.out.println(f2);
    f2++;
}
//    打优丁:8.4552632E7   納尼僻孝?不是 +1了嗎?
//    打邮匚健:8.4552632E7   納尼皮璧?不是 +1了嗎?
//    打臃址伞:8.4552632E7   納尼?不是 +1了嗎睹限?
//    打悠┟ā:8.4552632E7   納尼?不是 +1了嗎羡疗?
//    打尤痉:8.4552632E7   納尼?不是 +1了嗎叨恨?
//    打恿巍:8.4552632E7   納尼?不是 +1了嗎?
//    打颖拧:8.4552632E7   納尼痢毒?不是 +1了嗎?
//    打硬仙:8.4552632E7   納尼哪替?不是 +1了嗎?
//    打庸交场:8.4552632E7   納尼凭舶?不是 +1了嗎?
//    打影怠:8.4552632E7   納尼帅霜?不是 +1了嗎?

看到?jīng)]呼伸,這些簡單場景下的使用情況都很難滿足我們的需求身冀,所以說用浮點(diǎn)數(shù)(包括doublefloat)處理問題有非常多隱晦的坑在等著咱們!

怪不得技術(shù)總監(jiān)發(fā)狠話:誰要是敢在處理諸如 商品金額蜂大、訂單交易闽铐、以及貨幣計(jì)算時(shí)用浮點(diǎn)型數(shù)據(jù)(double/float),直接讓我們走人奶浦!

image

原因出在哪里兄墅?

我們就以第一個(gè)典型現(xiàn)象為例來分析一下:

System.out.println( 1f == 0.99999999f );

直接用代碼去比較10.99999999,居然打印出true澳叉!

image

這說明了什么隙咸?這說明了計(jì)算機(jī)壓根區(qū)分不出來這兩個(gè)數(shù)。這是為什么呢成洗?

我們不妨來簡單思考一下:

我們知道輸入的這兩個(gè)浮點(diǎn)數(shù)只是我們?nèi)祟惾庋鬯吹降木唧w數(shù)值五督,是我們通常所理解的十進(jìn)制數(shù),但是計(jì)算機(jī)底層在計(jì)算時(shí)可不是按照十進(jìn)制來計(jì)算的瓶殃,學(xué)過基本計(jì)組原理的都知道充包,計(jì)算機(jī)底層最終都是基于像010100100100110011011這種01二進(jìn)制來完成的遥椿。

所以為了搞懂實(shí)際情況基矮,我們應(yīng)該將這兩個(gè)十進(jìn)制浮點(diǎn)數(shù)轉(zhuǎn)化到二進(jìn)制空間來看一看。

十進(jìn)制浮點(diǎn)數(shù)轉(zhuǎn)二進(jìn)制 怎么轉(zhuǎn)冠场、怎么計(jì)算家浇,我想這應(yīng)該屬于基礎(chǔ)計(jì)算機(jī)進(jìn)制轉(zhuǎn)換常識(shí),在 《計(jì)算機(jī)組成原理》 類似的課上肯定學(xué)過了碴裙,咱就不在此贅述了钢悲,直接給出結(jié)果(把它轉(zhuǎn)換到IEEE 754 Single precision 32-bit点额,也就float類型對應(yīng)的精度)

1.0(十進(jìn)制) 
    ↓
00111111 10000000 00000000 00000000(二進(jìn)制) 
    ↓
0x3F800000(十六進(jìn)制)
0.99999999(十進(jìn)制) 
    ↓
00111111 10000000 00000000 00000000(二進(jìn)制) 
    ↓
0x3F800000(十六進(jìn)制)

果不其然,這兩個(gè)十進(jìn)制浮點(diǎn)數(shù)的底層二進(jìn)制表示是一毛一樣的莺琳,怪不得==的判斷結(jié)果返回true还棱!

但是1f == 0.9999999f返回的結(jié)果是符合預(yù)期的,打印false芦昔,我們也把它們轉(zhuǎn)換到二進(jìn)制模式下看看情況:

1.0(十進(jìn)制) 
    ↓
00111111 10000000 00000000 00000000(二進(jìn)制) 
    ↓
0x3F800000(十六進(jìn)制)
0.9999999(十進(jìn)制) 
    ↓
00111111 01111111 11111111 11111110(二進(jìn)制) 
    ↓
0x3F7FFFFE(十六進(jìn)制)

哦诱贿,很明顯,它倆的二進(jìn)制數(shù)字表示確實(shí)不一樣咕缎,這是理所應(yīng)當(dāng)?shù)慕Y(jié)果珠十。

那么為什么0.99999999的底層二進(jìn)制表示竟然是:00111111 10000000 00000000 00000000 呢?

這不明明是浮點(diǎn)數(shù)1.0的二進(jìn)制表示嗎凭豪?

這就要談一下浮點(diǎn)數(shù)的精度問題了焙蹭。


浮點(diǎn)數(shù)的精度問題!

學(xué)過 《計(jì)算機(jī)組成原理》 這門課的小伙伴應(yīng)該都知道嫂伞,浮點(diǎn)數(shù)在計(jì)算機(jī)中的存儲(chǔ)方式遵循IEEE 754 浮點(diǎn)數(shù)計(jì)數(shù)標(biāo)準(zhǔn)孔厉,可以用科學(xué)計(jì)數(shù)法表示為:

image

只要給出:符號(hào)(S)階碼部分(E)帖努、尾數(shù)部分(M) 這三個(gè)維度的信息撰豺,一個(gè)浮點(diǎn)數(shù)的表示就完全確定下來了,所以floatdouble這兩種浮點(diǎn)數(shù)在內(nèi)存中的存儲(chǔ)結(jié)構(gòu)如下所示:

image
image

1拼余、符號(hào)部分(S)

0-正 1-負(fù)

2污桦、階碼部分(E)(指數(shù)部分)

  • 對于float型浮點(diǎn)數(shù),指數(shù)部分8位匙监,考慮可正可負(fù)凡橱,因此可以表示的指數(shù)范圍為-127 ~ 128
  • 對于double型浮點(diǎn)數(shù),指數(shù)部分11位亭姥,考慮可正可負(fù)稼钩,因此可以表示的指數(shù)范圍為-1023 ~ 1024

3、尾數(shù)部分(M)

浮點(diǎn)數(shù)的精度是由尾數(shù)的位數(shù)來決定的:

  • 對于float型浮點(diǎn)數(shù)达罗,尾數(shù)部分23位坝撑,換算成十進(jìn)制就是 2^23=8388608,所以十進(jìn)制精度只有6 ~ 7位粮揉;
  • 對于double型浮點(diǎn)數(shù)绍载,尾數(shù)部分52位,換算成十進(jìn)制就是 2^52 = 4503599627370496滔蝉,所以十進(jìn)制精度只有15 ~ 16

所以對于上面的數(shù)值0.99999999f,很明顯已經(jīng)超過了float型浮點(diǎn)數(shù)據(jù)的精度范圍塔沃,出問題也是在所難免的蝠引。


精度問題如何解決

所以如果涉及商品金額阳谍、交易值貨幣計(jì)算等這種對精度要求很高的場景該怎么辦呢螃概?

方法一:用字符串或者數(shù)組解決多位數(shù)問題

校招刷過算法題的小伙伴們應(yīng)該都知道矫夯,用字符串或者數(shù)組表示大數(shù)是一個(gè)典型的解題思路。

比如經(jīng)典面試題:編寫兩個(gè)任意位數(shù)大數(shù)的加法吊洼、減法训貌、乘法等運(yùn)算

這時(shí)候我們我們可以用字符串或者數(shù)組來表示這種大數(shù)冒窍,然后按照四則運(yùn)算的規(guī)則來手動(dòng)模擬出具體計(jì)算過程递沪,中間還需要考慮各種諸如:進(jìn)位借位综液、符號(hào)等等問題的處理款慨,確實(shí)十分復(fù)雜,本文不做贅述谬莹。

方法二:Java的大數(shù)類是個(gè)好東西

JDK早已為我們考慮到了浮點(diǎn)數(shù)的計(jì)算精度問題檩奠,因此提供了專用于高精度數(shù)值計(jì)算的大數(shù)類來方便我們使用。

在前文《不瞞你說附帽,我最近跟Java源碼杠上了》中說過埠戳,Java的大數(shù)類位于java.math包下:

image

可以看到,常用的BigIntegerBigDecimal就是處理高精度數(shù)值計(jì)算的利器蕉扮。

BigDecimal num3 = new BigDecimal( Double.toString( 0.1f ) );
BigDecimal num4 = new BigDecimal( Double.toString( 0.99999999f ) );
System.out.println( num3 == num4 );  // 打印 false

BigDecimal num1 = new BigDecimal( Double.toString( 0.2 ) );
BigDecimal num2 = new BigDecimal( Double.toString( 0.7 ) );

// 加
System.out.println( num1.add( num2 ) );  // 打诱浮:0.9

// 減
System.out.println( num2.subtract( num1 ) );  // 打印:0.5

// 乘
System.out.println( num1.multiply( num2 ) );  // 打勇浴:0.14

// 除
System.out.println( num2.divide( num1 ) );  // 打幼δ!:3.5

當(dāng)然了,像BigIntegerBigDecimal這種大數(shù)類的運(yùn)算效率肯定是不如原生類型效率高荚藻,代價(jià)還是比較昂貴的屋灌,是否選用需要根據(jù)實(shí)際場景來評(píng)估。


最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末应狱,一起剝皮案震驚了整個(gè)濱河市共郭,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌疾呻,老刑警劉巖除嘹,帶你破解...
    沈念sama閱讀 206,378評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異岸蜗,居然都是意外死亡尉咕,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門璃岳,熙熙樓的掌柜王于貴愁眉苦臉地迎上來年缎,“玉大人悔捶,你說我怎么就攤上這事〉ノ撸” “怎么了蜕该?”我有些...
    開封第一講書人閱讀 152,702評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀的道長洲鸠。 經(jīng)常有香客問我堂淡,道長,這世上最難降的妖魔是什么扒腕? 我笑而不...
    開封第一講書人閱讀 55,259評(píng)論 1 279
  • 正文 為了忘掉前任绢淀,我火速辦了婚禮,結(jié)果婚禮上袜匿,老公的妹妹穿的比我還像新娘更啄。我一直安慰自己,他們只是感情好居灯,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,263評(píng)論 5 371
  • 文/花漫 我一把揭開白布祭务。 她就那樣靜靜地躺著,像睡著了一般怪嫌。 火紅的嫁衣襯著肌膚如雪义锥。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,036評(píng)論 1 285
  • 那天岩灭,我揣著相機(jī)與錄音拌倍,去河邊找鬼。 笑死噪径,一個(gè)胖子當(dāng)著我的面吹牛柱恤,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播找爱,決...
    沈念sama閱讀 38,349評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼梗顺,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了车摄?” 一聲冷哼從身側(cè)響起寺谤,我...
    開封第一講書人閱讀 36,979評(píng)論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎吮播,沒想到半個(gè)月后变屁,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,469評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡意狠,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,938評(píng)論 2 323
  • 正文 我和宋清朗相戀三年粟关,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片环戈。...
    茶點(diǎn)故事閱讀 38,059評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡闷板,死狀恐怖获列,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情蛔垢,我是刑警寧澤,帶...
    沈念sama閱讀 33,703評(píng)論 4 323
  • 正文 年R本政府宣布迫悠,位于F島的核電站鹏漆,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏创泄。R本人自食惡果不足惜艺玲,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,257評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望鞠抑。 院中可真熱鬧饭聚,春花似錦、人聲如沸搁拙。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽箕速。三九已至酪碘,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間盐茎,已是汗流浹背兴垦。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評(píng)論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留字柠,地道東北人探越。 一個(gè)月前我還...
    沈念sama閱讀 45,501評(píng)論 2 354
  • 正文 我出身青樓,卻偏偏與公主長得像窑业,于是被迫代替她去往敵國和親钦幔。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,792評(píng)論 2 345