在iOS開(kāi)發(fā)中,和貨幣價(jià)格計(jì)算相關(guān)的谍倦,需要注意計(jì)算精度的問(wèn)題塞赂。即使只是兩位小數(shù),也會(huì)出現(xiàn)誤差昼蛀。使用float類(lèi)型運(yùn)算宴猾,是完全不夠的。所以我在網(wǎng)上來(lái)找尋答案,但是在百度找了好久,沒(méi)有發(fā)現(xiàn)一個(gè)好的解決方案,后來(lái)發(fā)現(xiàn)了NSDecimalNumber這個(gè)類(lèi),但是國(guó)內(nèi)搜索引擎上的資料用太少了叼旋。
先敘述下我遇到的問(wèn)題,我的服務(wù)器傳給我的是一個(gè)float的值,作為一個(gè)對(duì)外的庫(kù)仇哆,由于存在版本延續(xù),需要保留對(duì)外的flaot的類(lèi)型夫植,不改變API税产,選擇進(jìn)行內(nèi)部適配。
問(wèn)題引出
float a =0.01;
int b =99999999;
double c =0.0;
c = a*b;NSLog(@"%f",c);
//輸出結(jié)果為 1000000.000000
NSLog(@"%.2f",c);
//輸出結(jié)果為 1000000.00
//明顯不夠精確
在網(wǎng)上找到了一個(gè)國(guó)內(nèi)朋友的博客也遇到和我一樣的問(wèn)題,他嘗試了如下兩種解決方案
將float強(qiáng)制轉(zhuǎn)換為double
c = a*(double)b;NSLog(@"%f",c);//輸出結(jié)果? 999999.967648NSLog(@"%.2f",c);//輸出結(jié)果? 999999.97// 明顯已經(jīng)丟失精度
通過(guò)和NSString的轉(zhuǎn)換偷崩,將計(jì)算的原始數(shù)據(jù)轉(zhuǎn)換為純粹的double類(lèi)型的數(shù)據(jù),這樣的計(jì)算精度就可以達(dá)到要求了
NSString *objA = [NSString stringWithFormat:@"%.2f", a];NSString *objB = [NSString stringWithFormat:@"%.2f", (double)b];c = [objA doubleValue] * [objB doubleValue];NSLog(@"%.2f",c);//輸出結(jié)果? 999999.99
最終方案
NSString *decimalNumberMutiplyWithString(NSString *multiplierValue,NSString *multiplicandValue){
NSDecimalNumber *multiplierNumber = [NSDecimalNumber decimalNumberWithString:multiplierValue];
NSDecimalNumber *multiplicandNumber = [NSDecimalNumber decimalNumberWithString:multiplicandValue];
NSDecimalNumber *product = [multiplicandNumber decimalNumberByMultiplyingBy:multiplierNumber];
return [product stringValue];
}
NSLog(@"%@",decimalNumberMutiplyWithString([NSString stringWithFormat:@"%f",a], [NSString stringWithFormat:@"%d",b]));//輸出結(jié)果? 999999.99
NSDecimalNumber針對(duì) float double 精度丟失問(wèn)題(解決方法)
?NSDecimalNumberHandler *roundingBehavior = [NSDecimalNumberHandler decimalNumberHandlerWithRoundingMode:NSRoundPlain scale:6 raiseOnExactness:NO raiseOnOverflow:NO raiseOnUnderflow:NO raiseOnDivideByZero:NO];
? ? ? ? ? ? NSDecimalNumber *resultWalletAmt = [self.walletAmt decimalNumberByRoundingAccordingToBehavior:roundingBehavior];
? ? ? ? ? ? self.txtAmt.txtContent.text= [[resultWalletAmtdecimalNumberBySubtracting:gas]stringValue];
NSDecimalNumber 比較大小
NSDecimalNumber *change = [NSDecimalNumber decimalNumberWithString:[NSString stringWithFormat:@"%0.3f", [[coin.today change] floatValue]]];
? ? ? ? ? ? if(change){//![coin.today.last isEqualToString:coin.today.open] &&
? ? ? ? ? ? ? ? NSComparisonResultresult = [changecompare:@(0)];
? ? ? ? ? ? ? ? if(result ==NSOrderedAscending){
? ? ? ? ? ? ? ? ? ? self.txtPrice.textColor=self.changeBgView.backgroundColor=kHex(@"E41856");
? ? ? ? ? ? ? ? ? ? self.txtChange.text= [NSStringstringWithFormat:@"%@%@%%",@"",change];
? ? ? ? ? ? ? ? }elseif(result ==NSOrderedDescending){
? ? ? ? ? ? ? ? ? ? self.txtPrice.textColor=self.changeBgView.backgroundColor=kHex(@"00A97E");
? ? ? ? ? ? ? ? ? ? self.txtChange.text= [NSStringstringWithFormat:@"%@%@%%",@"+",change];
? ? ? ? ? ? ? ? }else{
? ? ? ? ? ? ? ? ? ? self.txtPrice.textColor= [UIColorwhiteColor];
? ? ? ? ? ? ? ? ? ? self.changeBgView.backgroundColor=kHex(@"00A97E");
? ? ? ? ? ? ? ? ? ? self.txtChange.text=@"+0.00%";
? ? ? ? ? ? ? ? }
? ? ? ? ? ? }else{
? ? ? ? ? ? ? ? self.txtPrice.textColor = [UIColor whiteColor];
? ? ? ? ? ? ? ? self.changeBgView.backgroundColor=kHex(@"00A97E");
? ? ? ? ? ? ? ? self.txtChange.text=@"+0.00%";
? ? ? ? ? ? }
四舍五入
// 四舍五入NSDecimalNumberHandler*roundPlain = [NSDecimalNumberHandlerdecimalNumberHandlerWithRoundingMode:NSRoundPlainscale:2raiseOnExactness:NOraiseOnOverflow:NOraiseOnUnderflow:NOraiseOnDivideByZero:YES];? ? {NSString*addStr =@"1.0";NSString*oneStr =@"1.114";NSString*twoStr =@"1.116";NSDecimalNumber*oneNum = [NSDecimalNumberdecimalNumberWithString:oneStr];NSDecimalNumber*twoNum = [NSDecimalNumberdecimalNumberWithString:twoStr];NSDecimalNumber*addNum = [NSDecimalNumberdecimalNumberWithString:addStr];// 加法oneNum = [oneNum decimalNumberByAdding:addNum withBehavior:roundPlain];? ? ? ? twoNum = [twoNum decimalNumberByAdding:addNum withBehavior:roundPlain];NSLog(@"\n1.114 + 1.0 (保留兩位小數(shù))四舍五入的結(jié)果為:%@\n1.116 + 1.0 (保留兩位小數(shù))四舍五入的結(jié)果為:%@", oneNum, twoNum);? ? }