floatValue 失真

开发遇到的一个坑。在与后台交互的时候,协议规定的数据类型是浮点型,这就要求我们将用户输入的字符串转化为float类型。但是在转化过程中会出现失真的情况。
其实float与str互转会出现失真的情况,原因很简单。这是因为浮点数的存储方式的缘故。

    NSLog(@"%f",[@"20.5" floatValue]);
    NSLog(@"%f",[@"20.7" floatValue]);
    NSLog(@"%f",[@"20.75" floatValue]);
    NSLog(@"%f",[@"20.76" floatValue]);
    NSLog(@"%f",[@"20.875" floatValue]);
    NSLog(@"%f",[@"20.878" floatValue]);
    NSLog(@"%f",[@"20.09375" floatValue]);
    NSLog(@"%f",[@"20.09335" floatValue]);
    NSLog(@"%f",[@"20.546875" floatValue]);
    NSLog(@"%f",[@"20.546878" floatValue]);

2018-08-25 11:21:58.833098+0800 floatValueDistortion[3523:408253] 20.500000
2018-08-25 11:21:58.833285+0800 floatValueDistortion[3523:408253] 20.700001
2018-08-25 11:21:58.833401+0800 floatValueDistortion[3523:408253] 20.750000
2018-08-25 11:21:58.833520+0800 floatValueDistortion[3523:408253] 20.760000
2018-08-25 11:21:58.833623+0800 floatValueDistortion[3523:408253] 20.875000
2018-08-25 11:21:58.834162+0800 floatValueDistortion[3523:408253] 20.878000
2018-08-25 11:21:58.834477+0800 floatValueDistortion[3523:408253] 20.093750
2018-08-25 11:21:58.834636+0800 floatValueDistortion[3523:408253] 20.093349
2018-08-25 11:21:58.834784+0800 floatValueDistortion[3523:408253] 20.546875
2018-08-25 11:21:58.834978+0800 floatValueDistortion[3523:408253] 20.546879

在回顾了浮点数的存储方式之后,和后台商议使用更改数据类型为字符串进行传输,问题就解决了。但是如果要用输入的类型进行计算呢?这个时候问题就避不开了。
其实苹果已有线程的API是可以支持的。NSDecimalNumber看定义就可以发现,不同于浮点数的存储,他是十进制的。同时也可以定制精度的取正类型:向上取正、向下去正、四舍五入等。相对与浮点类型的计算,NSDecimalNumber提供了更加精准的计算。

NSDecimalNumber
An object for representing and performing arithmetic on base-10 numbers.
- (void)caculateTest
{
    NSString * floatStr1 =@"14.326";
    NSString * floatStr2 =@"7.54";
    
    NSDecimalNumber * num1 = [NSDecimalNumber decimalNumberWithString:floatStr1];
    NSDecimalNumber * num2 = [NSDecimalNumber decimalNumberWithString:floatStr2];

    NSDecimalNumber *num3 = [num1 decimalNumberByAdding:num2];
    CGFloat float3 = [floatStr1 floatValue] + [floatStr2 floatValue];
    NSLog(@"%@,%f",num3,float3);
    
    NSDecimalNumber *num4 = [num1 decimalNumberBySubtracting:num2];
    CGFloat float4 = [floatStr1 floatValue] - [floatStr2 floatValue];
    NSLog(@"%@,%f",num4,float4);

    NSDecimalNumber *num5 = [num1 decimalNumberByMultiplyingBy:num2];
    CGFloat float5 = [floatStr1 floatValue] * [floatStr2 floatValue];
    NSLog(@"%@,%f",num5,float5);

    NSDecimalNumber *num6 = [num1 decimalNumberByDividingBy:num2];
    CGFloat float6 = [floatStr1 floatValue] / [floatStr2 floatValue];
    NSLog(@"%@,%f",num6,float6);
}

floatValueDistortion[3897:448522] 21.866,21.866001
2018-08-25 12:31:09.503985+0800 floatValueDistortion[3897:448522] 6.786,6.786000
2018-08-25 12:31:09.504800+0800 floatValueDistortion[3897:448522] 108.01804,108.018044
2018-08-25 12:31:09.504993+0800 floatValueDistortion[3897:448522] 1.9,1.900000

Demo地址

浮点数在计算机中存储方式
iOS开发笔记之四十二——一个double精度导致的bug

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 1.现象: 因APP业务需要,从后台端取得轨迹点的string,包含一千及以上个经纬度坐标点,数值精度为小数点后六...
    Masazumi柒阅读 7,013评论 0 0
  • iOS开发中,接受后台的响应,把json然后转化为模型对象,最终转化为NSString对象。 后台返回的一个数字(...
    海边漫步的我阅读 6,017评论 0 4
  • 一、Java 简介 Java是由Sun Microsystems公司于1995年5月推出的Java面向对象程序设计...
    子非鱼_t_阅读 9,871评论 1 44
  • 同级别标签对齐; 嵌套标签缩进 2 个空格; 选择器和 “{” 之间加个空格; 属性名称: 属性值 之间的冒号...
    辉夜乀阅读 1,859评论 0 0
  • 【写在前面的絮絮叨叨】 - 文梗来源于平行宇宙的解释,其实没有什么关系,只是突然脑洞大开有了这个莫名其妙的文梗,总...
    Grey夏阅读 12,918评论 3 34

友情链接更多精彩内容