美文网首页ios
Objective-C货币的精确计算

Objective-C货币的精确计算

作者: 瞬csr | 来源:发表于2016-09-01 02:17 被阅读201次

    问题来源

    在iOS开发中,和货币价格计算相关的,是需要注意计算精度的问题的。即使只是两位小数,也会出现误差。使用float类型运算,是完全不够的。 如: 原始数据:

    float a = 0.01;
    int b = 99999999;
    double c = 0.0;
    

    1:使用浮点运算

    c = a*b;
    NSLog(@"%f",c);
    NSLog(@"%.2f",c);
    

    输出结果:

    2016-09-01 00:42:43.216 MoneyCalculateTest[1367:35625] ------1000000.000000
    2016-09-01 00:42:43.216 MoneyCalculateTest[1367:35625] ------1000000.00
    

    可见使用浮点型来计算是有误差的。

    2.使用类型转换,提高精度

    c = a*(double)b;
    NSLog(@"%f",c);
    NSLog(@"%.2f",c);
    

    输出:

    2016-09-01 00:42:43.216 MoneyCalculateTest[1367:35625] 999999.967648
    2016-09-01 00:42:43.216 MoneyCalculateTest[1367:35625] 999999.97
    

    Double运算的精度是提高了,可是浮点数的数值早已经出现了精度的不准确,即使存储空间足够,同样还是不准确的数值。

    3.通过和NSString的转换,将计算的原始数据转换为纯粹的double类型的数据。

    NSString *objA = [NSString stringWithFormat:@"%.2f", a];
    NSString *objB = [NSString stringWithFormat:@"%.2f", (double)b];
    c = [objA doubleValue] * [objB doubleValue];NSLog(@"%.2f",c);
    

    输出:

    2016-09-01 00:42:43.217 MoneyCalculateTest[1367:35625] 99999999.00
    

    计算的结果还是比较准确的,不过需要做格式化输入和格式化输出的处理。同时使用NSString来转换,这样的写法看起来比较奇怪,但是精确度已经可以达到要求了。

    系统提供的计算API

    事实上, 除了使用字符串来转换提高精确度,系统已经为我们封装好了一套专门用于计算的API,通过NSDecimalNumber提供的计算方式,可以很好的计算出准确的精度的数据,同时不需要使用格式化输出等。其计算的精度是比较高的,这是官方建议的货币计算的API,对乘除等计算都有单独的API接口来提供。

    可以将以下封装好的方法写在NString的类别中,这样就可以很方便的调用。

    1.加

    + (NSString *)addNumberMutiplyWithString:(NSString *)multiplierValue secondString:(NSString *)multiplicandValue
    { 
      NSDecimalNumber *multiplierNumber = [NSDecimalNumber decimalNumberWithString:multiplierValue]; 
      NSDecimalNumber *multiplicandNumber = [NSDecimalNumber decimalNumberWithString:multiplicandValue]; 
      NSDecimalNumber *product = [multiplicandNumber decimalNumberByAdding:multiplierNumber];  
    return [product stringValue];
    }
    

    2.减法
    + (NSString *)subNumberMutiplyWithString:(NSString *)multiplierValue secondString:(NSString *)multiplicandValue
    {
    NSDecimalNumber *multiplierNumber = [NSDecimalNumber decimalNumberWithString:multiplierValue];
    NSDecimalNumber *multiplicandNumber = [NSDecimalNumber decimalNumberWithString:multiplicandValue];
    NSDecimalNumber *product = [multiplicandNumber decimalNumberBySubtracting:multiplierNumber];
    return [product stringValue];
    }
    3.乘法
    + (NSString *)mutiplyNumberMutiplyWithString:(NSString *)multiplierValue secondString:(NSString *)multiplicandValue
    {
    NSDecimalNumber *multiplierNumber = [NSDecimalNumber decimalNumberWithString:multiplierValue];
    NSDecimalNumber *multiplicandNumber = [NSDecimalNumber decimalNumberWithString:multiplicandValue];
    NSDecimalNumber *product = [multiplicandNumber decimalNumberByMultiplyingBy:multiplierNumber];
    return [product stringValue];
    }

    4.除法

    + (NSString *)divisionNumberMutiplyWithString:(NSString *)multiplierValue secondString:(NSString *)multiplicandValue
    { 
     NSDecimalNumber *multiplierNumber = [NSDecimalNumber decimalNumberWithString:multiplierValue];
     NSDecimalNumber *multiplicandNumber = [NSDecimalNumber decimalNumberWithString:multiplicandValue];
     NSDecimalNumber *product = [multiplicandNumber decimalNumberByDividingBy:multiplierNumber];  
     return [product stringValue];
    }
    

    5.简单的应用

    - (void)viewDidLoad
    {
      [super viewDidLoad];
            
      float a = 0.01;
      int b = 99999999;
      double c = 0.0;
    
      //调用第一种方法来计算
      NSString *str =[self mutiplyNumberMutiplyWithString:[NSString stringWithFormat:@"%f",a] secondString:[NSString stringWithFormat:@"%d",b]];
      NSLog(@"%@",str);
      c = [str doubleValue];
    
      //调用第二种方法
      NSLog(@"%@",decimalNumberMutiplyWithString([NSString stringWithFormat:@"%f",a], [NSString stringWithFormat:@"%d",b]));
            
    }
    //乘法的封装方法一
    - (NSString *)mutiplyNumberMutiplyWithString:(NSString *)multiplierValue secondString:(NSString *)multiplicandValue
    {
       NSDecimalNumber *multiplierNumber = [NSDecimalNumber decimalNumberWithString:multiplierValue];
       NSDecimalNumber *multiplicandNumber = [NSDecimalNumber decimalNumberWithString:multiplicandValue];
       NSDecimalNumber *product = [multiplicandNumber decimalNumberByMultiplyingBy:multiplierNumber];
            
       return [product stringValue];
    }
    //第二种封装方法,可以直接像函数一样调用该方法
    NSString *decimalNumberMutiplyWithString(NSString *multiplierValue,NSString *multiplicandValue)
    {       
       NSDecimalNumber *multiplierNumber = [NSDecimalNumber decimalNumberWithString:multiplierValue];
       NSDecimalNumber *multiplicandNumber = [NSDecimalNumber decimalNumberWithString:multiplicandValue];
       NSDecimalNumber *product = [multiplicandNumber decimalNumberByMultiplyingBy:multiplierNumber];
       return [product stringValue];          
    }
    

    输出结果

    2016-09-01 02:05:57.714 MoneyCalculateTest[2031:89967] 999999.99
    2016-09-01 02:05:57.715 MoneyCalculateTest[2031:89967] 999999.99
    

    精确度十分OK,因此当我们进行数值计算的时候,优先考虑使用系统的方法来进行计算。

    相关文章

      网友评论

        本文标题:Objective-C货币的精确计算

        本文链接:https://www.haomeiwen.com/subject/ifecettx.html