大家都会在开发中遇到, float或者 double 类型有时候会有一个精度的丢失情况, 造成数据不准确,或者导致UI出现问题, 尤其是在涉及到一些支付和金钱展示的项目, 都要求比较精确。
所以苹果在这背景下推出了NSDecimalNumber 类 来优化数值的精准度问题。
NSDecimalNumber 的简单介绍
1. NSDecimalNumber 继承自 NSNumber
首先NSDecimalNumber 继承自 NSNumber,所以自然拥有 NSNumber 的相关属性和方法,需要注意的是 NSNumber 继承自 NSValue 在复杂的数据处理的时候知道这些会快速解决问题。
2. NSDecimalNumber的精确度
通过NSDecimalNumber和NSDecimalNumberHandler的使用可以快速的对数据处理。
屏幕快照 2019-05-15 21.15.00.png参数说明:
decimalNumberHandlerWithRoundingMode:对数据处理的方式
scale:小数点后几位
根据官方文档说明, 枚举值分析:
NSRoundPlain, 四舍五入
NSRoundDown, 只舍不入
NSRoundUp, 只入不舍
NSRoundBankers 四舍六入, 中间值时, 取最近的,保持保留最后一位为偶数
对数据四舍五入的封装:
-(NSString *)notRounding:(float)price afterPoint:(int)position{
NSDecimalNumberHandler* roundingBehavior = [NSDecimalNumberHandler decimalNumberHandlerWithRoundingMode:NSRoundDown scale:position raiseOnExactness:NO raiseOnOverflow:NO raiseOnUnderflow:NO raiseOnDivideByZero:NO];
NSDecimalNumber *ouncesDecimal;
NSDecimalNumber *roundedOunces;
ouncesDecimal = [[NSDecimalNumber alloc] initWithFloat:price];
roundedOunces = [ouncesDecimal decimalNumberByRoundingAccordingToBehavior:roundingBehavior];
return [NSString stringWithFormat:@"%@",roundedOunces];
}
参数说明:
price:要处理的值
position:四舍五入的小数点后几位
只需要改变decimalNumberHandlerWithRoundingMode的枚举就行了。
3. NSDecimalNumber的处理数值计算
NSDecimalNumber 内部封装了 对数据处理的方法
// 加法
- (NSDecimalNumber *)decimalNumberByAdding:(NSDecimalNumber *)decimalNumber;
// 添加约束行为
- (NSDecimalNumber *)decimalNumberByAdding:(NSDecimalNumber *)decimalNumber withBehavior:(nullable id <NSDecimalNumberBehaviors>)behavior;
// 减法
- (NSDecimalNumber *)decimalNumberBySubtracting:(NSDecimalNumber *)decimalNumber;
- (NSDecimalNumber *)decimalNumberBySubtracting:(NSDecimalNumber *)decimalNumber withBehavior:(nullable id <NSDecimalNumberBehaviors>)behavior;
// 乘法
- (NSDecimalNumber *)decimalNumberByMultiplyingBy:(NSDecimalNumber *)decimalNumber;
- (NSDecimalNumber *)decimalNumberByMultiplyingBy:(NSDecimalNumber *)decimalNumber withBehavior:(nullable id <NSDecimalNumberBehaviors>)behavior;
// 除法
- (NSDecimalNumber *)decimalNumberByDividingBy:(NSDecimalNumber *)decimalNumber;
- (NSDecimalNumber *)decimalNumberByDividingBy:(NSDecimalNumber *)decimalNumber withBehavior:(nullable id <NSDecimalNumberBehaviors>)behavior;
例如:两个float类型的计算
CGFloat oneNum = 1.222;
CGFloat twoNum = 1.223;
NSDecimalNumber *decOneNum = [[NSDecimalNumber alloc]initWithFloat:oneNum];
NSDecimalNumber *decTwoNum = [[NSDecimalNumber alloc]initWithFloat:twoNum];
// 四舍五入 小数点后两位
NSDecimalNumberHandler *roundingBehavior = [NSDecimalNumberHandler decimalNumberHandlerWithRoundingMode:NSRoundPlain scale:2.0f raiseOnExactness:NO raiseOnOverflow:NO raiseOnUnderflow:NO raiseOnDivideByZero:NO];
NSDecimalNumber *resultNum = [decOneNum decimalNumberByDividingBy:decTwoNum withBehavior:roundingBehavior];
其他算法类似。
4. NSDecimalNumber处理数据的比较
通过下面的方法实现数值的比较:
- (NSComparisonResult)compare:(NSNumber *)decimalNumber;
比较完成之后会返回 NSComparisonResult 值,它是一个枚举类型
typedef NS_CLOSED_ENUM(NSInteger, NSComparisonResult) {
NSOrderedAscending = -1L, 大于
NSOrderedSame, 等于
NSOrderedDescending 小于
};
例如:
NSDecimalNumber *decOneNum = [[NSDecimalNumber alloc]initWithFloat:oneNum];
NSDecimalNumber *decTwoNum = [[NSDecimalNumber alloc]initWithFloat:twoNum];
switch (result) {
case NSOrderedAscending:
return @"小于";
break;
case NSOrderedSame:
return @"等于";
break;
case NSOrderedDescending:
return @"大于";
break;
default:
return @"";
break;
}
我对上面的方法进行了封装直接调用就好 地址:
这样我们对数据的处理就方便多了。
网友评论