先看現(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ù)(包括double
和float
)處理問題有非常多隱晦的坑在等著咱們!
怪不得技術(shù)總監(jiān)發(fā)狠話:誰要是敢在處理諸如 商品金額蜂大、訂單交易闽铐、以及貨幣計(jì)算時(shí)用浮點(diǎn)型數(shù)據(jù)(double
/float
),直接讓我們走人奶浦!
原因出在哪里兄墅?
我們就以第一個(gè)典型現(xiàn)象為例來分析一下:
System.out.println( 1f == 0.99999999f );
直接用代碼去比較1
和0.99999999
,居然打印出true
澳叉!
這說明了什么隙咸?這說明了計(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
這種0
、1
二進(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ù)法表示為:
只要給出:符號(hào)(S)、階碼部分(E)帖努、尾數(shù)部分(M) 這三個(gè)維度的信息撰豺,一個(gè)浮點(diǎn)數(shù)的表示就完全確定下來了,所以float
和double
這兩種浮點(diǎn)數(shù)在內(nèi)存中的存儲(chǔ)結(jié)構(gòu)如下所示:
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
包下:
可以看到,常用的BigInteger
和 BigDecimal
就是處理高精度數(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)然了,像BigInteger
和 BigDecimal
這種大數(shù)類的運(yùn)算效率肯定是不如原生類型效率高荚藻,代價(jià)還是比較昂貴的屋灌,是否選用需要根據(jù)實(shí)際場景來評(píng)估。