iOS开发可能遇到的坑(一)——浮点数的转化显示问题

之前在学习C语言的时候课堂上,老师就强调,不能使用float类型的数字进行相等比较判断。这个也确实听进去了,也明白原因是float是存储是不精确的。但是真正开发实践的时候,或许只有出问题了,才会醒悟:哦,原来是这样。这个问题在高大上的OC上同样存在,稍不注意就会出现问题。

iOS开发中,接受后台的响应,然后转化为模型对象,最终转化为NSString对象,然后控件显示出来。这一切都是那么的自然那么的熟悉。
一个数字,可以定义为number类型,也可以定义为字符串问题。如果后台返回的是字符串类型。在iOS json序列化的时候,会把字符串类型转化为NSString对象,这个一点问题没有。但是如果后台返回的是number类型。json序列化会将number类型转化为NSNumber对象。使用的时候,想当然的会将NSNumber转化为NSString对象。这样做很自然啊,没有问题,也用一两个数字测试了,转化是精确的。测试那边也测试通过了,然后产品上线了。最终,还是出现了问题。不说大数,就10以内的吧。有这么多的转换不精确。(如下图)不信你可以试一下,让后台定义double类型数据66.6,你转化为字符串会发现就是66.59999999999999。测试发现double转化为NSNumber的时候就会出现问题。double表示的字面值和其存储的值一般是不一样的,转化为NSNumber之后,有很大一些数字,在NSNumber的stringValue上会出错。

  NSString* json = @"{\"number\":66.6}";
  NSData* data = [json dataUsingEncoding:NSUTF8StringEncoding];
  NSDictionary* dic = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableLeaves error:nil];
  id object = dic[@"number"];
  NSLog(@"%@",[object class]);
  NSLog(@"%@",object);

打印结果

 __NSCFNumber
 66.59999999999999
示例

解决方法可以是把取String的doubleValue,然后再.2f保留两位小数进行构造字符串。千万不能取floatValue,floatValue在大于15万的浮点数字就会出现不精确了(笔者做过遍历测试)。而doubleValue在数十亿的范围内都是字面上精确的。
根本解决方法还是劝后台的同事把请求返回的数据全部设置为String类型
如果涉及到计算的问题,那就只得乖乖转化为NSDecimalNumber对象了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 第5章 引用类型(返回首页) 本章内容 使用对象 创建并操作数组 理解基本的JavaScript类型 使用基本类型...
    大学一百阅读 3,280评论 0 4
  • ios开发中,在接收后台数据的时候会先转换为对象,最后转化为我们想要的数据类型. 那么数字是怎么样的一个过程呢??...
    香橙柚子阅读 3,585评论 1 2
  • 芦苇荡轻舟,浪花惊海鸥。 烤鱼虾,柳叶水边。 仙鹤欢歌千飞逸,忆难却,醉思乡。 魂渡又万里,几封故旧书。 念村庄,...
    雨意生香阅读 532评论 7 11
  • 第二天周六,早上7点,陆慕言起床,洗漱,换了一条黑色休闲西裤,露出脚踝,搭配白色衬衫和白色板鞋,收拾完房间,拿起床...
    莫穴阅读 322评论 0 2
  • 遐迩一体的意思是无论是边远地区的异族,还是内地的人民,都能够紧密团结,融为一体。 遐指的是远方,迩指的是附近,我们...
    曹斐阅读 1,039评论 0 4