我想将持续时间存储在 Double
中,并且需要提取 Double
的高值和低值(小时和分钟)以访问它们。我还需要能够使用两个 Ints
(hour
和 minute
)设置 Double
。
我现在在这里得到了一些东西:
extension Double {
func hour() -> Int {
return NSNumber(double: floor(self)).integerValue
}
func minute() -> Int {
return NSNumber(float: Float(Double((self - floor(self)) * 100))).integerValue
}
init(hour: Int, minute: Int) {
self = Double(hour) + ( Double(minute) / 100 )
}
}
然后我在我的 appDelegate 的 didFinishLaunchingWithOptions
中对此进行了一些测试:
var time: Double = 18.30
NSLog("Time is \"%f\" - so hours is \"%d\" and minutes is \"%d\".", time, time.hour(), time.minute())
time = Double(hour: 10, minute: 1)
NSLog("Time is \"%f\" - so hours is \"%d\" and minutes is \"%d\".", time, time.hour(), time.minute())
time = Double(hour: 15, minute: 45)
NSLog("Time is \"%f\" - so hours is \"%d\" and minutes is \"%d\".", time, time.hour(), time.minute())
现在,这段代码可以工作了。有人可能想知道为什么我在 minute()
函数中将 Double
转换为 Float
。这是因为在将这些 Float
转换为 之前,我在使用
。如果分钟未以 Double
而未将它们转换为 Float
时遇到了一个未知问题整数0
结尾,则由于未知原因减去一分钟。
例如,10:42 返回 10 小时 和 41 分钟。
但是 10:30 仍然是 10 小时 30 分钟。< br/>另外 10:00 是 10 小时 和 0 分钟。
我在这里发布的这个例程似乎有效,但我只是想知道,如果我尽可能高效......这里的转换比需要的多。
那么,有什么改进吗?
有人知道我提到的分钟减少的未知问题的原因吗?
最佳答案
二进制 float ,例如 Double
或 Float
不能准确表示数字 10.42
,因此
将小数部分乘以 100 并截断结果可能会得到 41
。
通过转换为 Float
,您只是在“隐藏”问题。
这是一个舍入而不是截断的版本,总体上更简单一些。我从你的代码中采用了“分钟”的符号, 即使它代表一小时的 1/100 而不是 1/60。
extension Double {
func hour() -> Int {
return Int(self)
}
func minute() -> Int {
return Int(round(self * 100.0)) % 100
}
init(hour: Int, minute: Int) {
self = Double(hour) + ( Double(minute) / 100 )
}
}
解决您问题的其他可能方法:
将持续时间存储为代表总计的整数 分钟数。例如。 “10:42”将存储为
10*60 + 42
。 那么你就完全没有舍入问题。使用基金会 输入
NSDecimalNumber
而不是Double
,它可以表示 最多 38 位的十进制整数。
关于ios - Swift 中 Double 的拆分和组合,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28254510/