6

我有一个应用程序使用 NSTimer 以厘秒(0.01 秒)的更新间隔将正在运行的秒表以字符串格式显示为 00:00.00 (mm:ss.SS)。(基本克隆了iOS内置秒表集成到实时运动计时数学题中,未来可能需要毫秒精度)

我使用(滥用?)NSTimer 来强制更新 UILabel。如果用户按下 Start,这是用于开始重复该功能的 NSTimer 代码:

displayOnlyTimer = NSTimer.scheduledTimerWithTimeInterval(0.01, target: self, selector: Selector("display"), userInfo: nil, repeats: true)

这是上面 NSTimer 执行的函数:

func display() {
    let currentTime = CACurrentMediaTime() - timerStarted + elapsedTime

    if currentTime < 60 {
        timeDisplay.text = String(format: "%.2f", currentTime)
    }else if currentTime < 3600 {
        var minutes = String(format: "%00d", Int(currentTime/60))
        var seconds = String(format: "%05.2f", currentTime % 60)
        timeDisplay.text =  minutes + ":" + seconds
    }else {
        var hours = String(format: "%00d", Int(currentTime/3600))
        var minutes = String(format: "%02d", (Int(currentTime/60)-(Int(currentTime/3600)*60)))
        var seconds = String(format: "%05.2f", currentTime % 60)
        timeDisplay.text =  hours + ":" + minutes + ":" + seconds
    }
}

将至少有 2 个显示链接同时运行。一旦所有其他元素都在起作用,这种方法会不会太低效?

当用户按下停止/暂停/重置时,显示会在不使用 NSTimer 的情况下更新。我没有找到任何直接翻译成 Swift 的东西。我相当肯定我正在使用一种低效的方法来强制在 UIView 中快速更新文本 UILabel。

更多详细信息:我正在为正在运行的计时器格式 (mm:ss.SS) 编写不那么混乱的代码。完成后我将再次更新。

更新:感谢 Rob 和 jtbandes 回答我的两个问题(格式化方法和显示更新方法)。用 CADisplayLink() 替换 NSTimer(见上文)很容易:

displayLink = CADisplayLink(target: self, selector: Selector("display"))
        displayLink.addToRunLoop(NSRunLoop.currentRunLoop(), forMode: NSRunLoopCommonModes)

然后替换代码中的所有实例

displayOnlyTimer.invalidate() 

displayLink.paused = true 

(这将暂停显示链接的更新)

4

2 回答 2

7

对于快速 UI 更新,您应该使用CADisplayLink。任何比显示刷新率快的东西都是浪费处理能力,因为它实际上无法显示。它还提供了timestamp前一帧的信息,因此您可以尝试预测下一帧的时间。

您正在计算CACurrentMediaTime() - timerStarted + elapsedTime多次。我建议只做一次并将其保存在局部变量中。

考虑使用NSDateComponentsFormatter。尝试重用格式化程序的一个实例,而不是每次都创建一个新实例(这通常是最昂贵的部分)。总的来说,你可以做的字符串操作越少越好。

您可以在显示方法的开头和结尾检查 CACurrentMediaTime 以查看需要多长时间。理想情况下,它应该小于 16.6 毫秒。密切关注 Xcode 调试导航器中的 CPU 使用率(和一般功耗)。

于 2015-07-13T04:29:35.990 回答
1

我今天正在解决同样的问题并找到了这个答案。Rob 和 jtbandes 的建议很有帮助,我能够从互联网上收集干净且有效的解决方案。谢谢你们。感谢 mothy 的提问。

我决定使用 CADisplayLink 因为触发计时器的回调比屏幕更新更频繁是没有意义的:

class Stopwatch: NSObject {
    private var displayLink: CADisplayLink!
    //...

    override init() {
        super.init()

        self.displayLink = CADisplayLink(target: self, selector: "tick:")
        displayLink.paused = true
        displayLink.addToRunLoop(NSRunLoop.mainRunLoop(), forMode: NSRunLoopCommonModes)
        //...
    }
    //...
}

我通过 displayLink.duration 每个刻度增加 elapsedTime 变量来跟踪时间:

var elapsedTime: CFTimeInterval!

override init() {
    //...
    self.elapsedTime = 0.0
    //...
}

func tick(sender: CADisplayLink) {
    elapsedTime = elapsedTime + displayLink.duration
    //...
}

时间格式化是通过 NSDateFormatter 完成的:

private let formatter = NSDateFormatter()

override init() {
//...
        formatter.dateFormat = "mm:ss,SS"
}

func elapsedTimeAsString() -> String {
        return formatter.stringFromDate(NSDate(timeIntervalSinceReferenceDate: elapsedTime))
}

UI 可以在 Stopwatch 每次滴答时调用的回调闭包中更新:

var callback: (() -> Void)?

func tick(sender: CADisplayLink) {
    elapsedTime = elapsedTime + displayLink.duration

    // Calling the callback function if available
    callback?()
}

这就是您在 ViewController 中使用秒表所需要做的一切:

let stopwatch = Stopwatch()
stopwatch.callback = self.tick

func tick() {
   elapsedTimeLabel.text = stopwatch.elapsedTimeAsString()
}

这是秒表和使用指南的完整代码的要点: https ://gist.github.com/Flar49/06b8c9894458a3ff1b14

我希望这个解释和要点能帮助其他人在未来偶然发现这个线程有同样的问题:)

于 2015-11-07T18:58:01.740 回答