12

我研究了stackoverflow,找到了我转换为Swift的解决方案,它似乎不起作用,并且选择器仍在执行中。

 func searchBar(searchBar: UISearchBar, textDidChange searchText: String) {
    self.filter.searchTerm = self.searchBar.text

    NSObject.cancelPreviousPerformRequestsWithTarget(self, selector: "getHints", object: nil)
    NSTimer.scheduledTimerWithTimeInterval(1, target: self, selector: "getHints", userInfo: nil, repeats: false)
}

有没有更好的方法可以快速做到这一点?谢谢!

4

1 回答 1

54

更新 2016/09/01:

我们可以使用NSTimers或(从 swift 2.0 开始)NSObjectperformSelector和朋友。

方法1:performSelector

func textField(_ textField: UITextField, shouldChangeCharactersIn range: NSRange,
    replacementString string: String) -> Bool {
    NSObject.cancelPreviousPerformRequests(
        withTarget: self,
        selector: #selector(ViewController.getHintsFromTextField),
        object: textField)
    self.perform(
        #selector(ViewController.getHintsFromTextField),
        with: textField,
        afterDelay: 0.5)
    return true
}

func getHintsFromTextField(textField: UITextField) {
    print("Hints for textField: \(textField)")
}

方法二:NSTimer

var timer: NSTimer? = nil

func textField(_ textField: UITextField, shouldChangeCharactersIn range: NSRange,
    replacementString string: String) -> Bool {
    timer?.invalidate()
    timer = Timer.scheduledTimer(
        timeInterval: 0.5,
        target: self,
        selector: #selector(ViewController.getHints),
        userInfo: ["textField": textField],
        repeats: false)
    return true
}

func getHints(timer: Timer) {
    var userInfo = timer.userInfo as! [String: UITextField]
    print("Hints for textField: \(userInfo["textField"])")
}

注意我将传递textField给延迟函数。它并不总是必需的,但在textField不容易访问或处理各种文本字段时,它可以让您的生活更轻松。

方法有什么NSTimer不同?performSelector

当您调用时performSelector,目标保留(在 swift 中,目标始终为self),但是当您使用NSTimer目标时,目标不会被保留。这意味着,如果您使用NSTimers,则必须确保目标(在本例self中为 )在计时器触发时是活动的。否则会发生崩溃。

(顺便说一句:内部performSelector使用NSTimer

如果您对 GCD 计时器感兴趣,请从这里开始: maicki/TimerWithGCD.md

于 2015-04-21T04:43:25.603 回答