NSDecimalNumber是使用貨幣時(shí)最好的格式。
官方文檔
NSDecimalNumber 是 NSNumber 的一個(gè)子類, 比 NSNumber 的功能更為強(qiáng)大, 四舍五入, 取整, 輸入后自動(dòng)去掉數(shù)值前面無(wú)用的 0 等等. 由于 NSDecimalNumber 精度較高, 所以會(huì)比基本數(shù)據(jù)類型費(fèi)時(shí), 所以需要權(quán)衡考慮, 蘋果官方建議在貨幣以及要求精度很高的場(chǎng)景下使用.
通常情況下我們會(huì)使用 NSDecimalNumberHandler 這個(gè)格式化器對(duì)其需要約束的格式進(jìn)行設(shè)置
NSDecimalNumberHandler
let decHandler:NSDecimalNumberHandler = NSDecimalNumberHandler.init(roundingMode: .plain, scale: 2, raiseOnExactness: false, raiseOnOverflow: false, raiseOnUnderflow: false, raiseOnDivideByZero: true)
roundingMode 文檔是一個(gè)枚舉類型
public enum RoundingMode : UInt {
case plain = 0 //只入不舍
case down = 1//只舍不入
case up = 2//四舍五入
case bankers = 3//四舍六入, 中間值時(shí), 取最近的,保持保留最后一位為偶數(shù)
}
scale: 保留幾位小數(shù)
raiseOnExactness: 發(fā)生精確錯(cuò)誤時(shí)是否拋出異常俗孝,一般為false
raiseOnOverflow: 發(fā)生溢出錯(cuò)誤時(shí)是否拋出異常,一般為false
raiseOnUnderflow: 發(fā)生不足錯(cuò)誤時(shí)是否拋出異常,一般為false
raiseOnDivideByZero: 除數(shù)是0時(shí)是否拋出異常搀绣,一般為true
補(bǔ)零問(wèn)題
NSDecimalNumber 做精度限制, 如果小數(shù)點(diǎn)后面是以0結(jié)尾, 則不會(huì)顯示,如1024.00 保留2位小數(shù) 結(jié)果為:1024
前端為了顯示效果需要補(bǔ)零操作
let decNumber = NSDecimalNumber.init(string: "1024")
let decHandler:NSDecimalNumberHandler = NSDecimalNumberHandler.init(roundingMode: .plain, scale: 2, raiseOnExactness: false, raiseOnOverflow: false, raiseOnUnderflow: false, raiseOnDivideByZero: true)
let dNumber = decNumber.rounding(accordingToBehavior: decHandler)
let p = 2
let count : Int = (p < 0 ? 0 : p)
var formatterString : String = "0."
if count == 0 {
formatterString = "0"
}
for _ in 0 ..< count {
formatterString.append("0")
}
let formatter : NumberFormatter = NumberFormatter()
formatter.positiveFormat = formatterString
formatter.string(from: dNumber)!
完整方法
func precisionString(with number:String, precision: String , isRound: Bool = false) -> String {
guard isPurnFloat(string: number) && isPurnFloat(string: precision) else {
return number
}
guard conerData(precision, .MyInt).0! >= 0 else {
return number
}
let decNumber = NSDecimalNumber.init(string: number)
var decHandler:NSDecimalNumberHandler?
if isRound {
decHandler = NSDecimalNumberHandler.init(roundingMode: .plain, scale: Int16(precision)!, raiseOnExactness: false, raiseOnOverflow: false, raiseOnUnderflow: false, raiseOnDivideByZero: true)
}else{
if conerData(number, .MyDouble).2! > 0 {
decHandler = NSDecimalNumberHandler.init(roundingMode: .up, scale: Int16(precision)!, raiseOnExactness: false, raiseOnOverflow: false, raiseOnUnderflow: false, raiseOnDivideByZero: true)
}else{
decHandler = NSDecimalNumberHandler.init(roundingMode: .up, scale: Int16(precision)!, raiseOnExactness: false, raiseOnOverflow: false, raiseOnUnderflow: false, raiseOnDivideByZero: true)
}
}
let dNumber = decNumber.rounding(accordingToBehavior: decHandler)
let p = conerData(precision, .MyInt).0!
let count : Int = (p < 0 ? 0 : p)
var formatterString : String = "0."
if count == 0 {
formatterString = "0"
}
for _ in 0 ..< count {
formatterString.append("0")
}
let formatter : NumberFormatter = NumberFormatter()
formatter.positiveFormat = formatterString
return formatter.string(from: dNumber)!
}
//MARK:- 是否數(shù)字
public func isPurnFloat(string: String) -> Bool {
let scan: Scanner = Scanner(string: string)
var val:Double = 0
return scan.scanDouble(&val) && scan.isAtEnd
}
NSDecimalNumber運(yùn)算
NSDecimalNumber:是蘋果針對(duì)浮點(diǎn)型計(jì)算時(shí)存在精度誤差的問(wèn)題而提供的一個(gè)類
let num1 = NSDecimalNumber.init(string: "1024.1024")
let num2 = NSDecimalNumber.init(string: "1024.1024")
//加
String(format: "%@", num1.adding(num2))
//減
String(format: "%@", num1.subtracting(num2))
//乘
String(format: "%@", num1.multiplying(num2))
//除
String(format: "%@", num1.dividing(num2))
//n次方(2次方)
String(format: "%@", num1.raising(toPower: 2))
//指數(shù)
String(format: "%@", num1.multiplying(byPowerOf10: 2))
相關(guān)文檔
open func adding(_ decimalNumber: NSDecimalNumber) -> NSDecimalNumber
open func adding(_ decimalNumber: NSDecimalNumber, withBehavior behavior: NSDecimalNumberBehaviors?) -> NSDecimalNumber
open func subtracting(_ decimalNumber: NSDecimalNumber) -> NSDecimalNumber
open func subtracting(_ decimalNumber: NSDecimalNumber, withBehavior behavior: NSDecimalNumberBehaviors?) -> NSDecimalNumber
open func multiplying(by decimalNumber: NSDecimalNumber) -> NSDecimalNumber
open func multiplying(by decimalNumber: NSDecimalNumber, withBehavior behavior: NSDecimalNumberBehaviors?) -> NSDecimalNumber
open func dividing(by decimalNumber: NSDecimalNumber) -> NSDecimalNumber
open func dividing(by decimalNumber: NSDecimalNumber, withBehavior behavior: NSDecimalNumberBehaviors?) -> NSDecimalNumber
open func raising(toPower power: Int) -> NSDecimalNumber
open func raising(toPower power: Int, withBehavior behavior: NSDecimalNumberBehaviors?) -> NSDecimalNumber
// compare two NSDecimalNumbers
open class var defaultBehavior: NSDecimalNumberBehaviors
// One behavior per thread - The default behavior is
// rounding mode: NSRoundPlain
// scale: No defined scale (full precision)
// ignore exactnessException
// raise on overflow, underflow and divide by zero.
open var objCType: UnsafePointer<CChar> { get }
// return 'd' for double
open var doubleValue: Double { get }
open func multiplying(byPowerOf10 power: Int16) -> NSDecimalNumber
open func multiplying(byPowerOf10 power: Int16, withBehavior behavior: NSDecimalNumberBehaviors?) -> NSDecimalNumber