天天看点

【iOS开发】@objc private class func 与 Selector

本文介绍在 Swift 中 使用 KVO 时,一些需要注意的事项,下面截取了两段我自己在项目中使用的代码。

self.timer = NSTimer.scheduledTimerWithTimeInterval(timeInterval, target: self,
        selector: "moveText", userInfo: nil, repeats: true)

NSNotificationCenter.defaultCenter().addObserver(self,
        selector: "appEnterForeground", name: nil, object: nil)
           

第一个是第一个定时器(NSTimer),第二个是一个页面的监听事件。即,让程序不断地观察现在的『某个状态』,当这个状态符合某个条件时,就去执行 Selector 中的代码。比如,我们的界面中有一个让用户输入年龄的 UITextField,然后我们可以观察(监听),当发现用户在这里输入『0』时,我们就给一个提示:『年龄至少为1』。

简单介绍完毕,下面说注意事项。

<h3>1. private 关键字</h3>

最简单的使用 Selector 的情况,就是如上面所列出的,在 NSTimer 中写 selector 为 "moveText",然后我们在同一个文件中,再相应地写一个

func moveText(){ 
    // Your Code
}
           

就可以了。

然而如果你想把它声明为一个 private func,即不希望其他文件调用这个函数,就会出现问题,通俗地讲:private 使得这个函数不能被『外部』调用,而 Selector 对它的调用,实际上就是一种『外部调用』。

【解决方案】

KVO 是基于 KVC 和动态派发技术(Dynamic Dispatch)的,Swift 为了效率禁用了 Objective-C 中有的动态派发。那么你把这个动态特性再加回来就可以了~

1.通过 @objc 关键字来使用动态特性

@objc private func moveText(){ 
    // Your Code
}
           

2.通过 dynamic 关键字来让这个函数拥有与 OC 中动态调用时相同的运行时特性(@objc 足矣,这个一般用不上)

dynamic private func moveText(){ 
    // Your Code
}
           

更多关于 dynamic 关键字的内容可以参考:

http://swifter.tips/kvo/

<h3>2. class 关键字</h3>

如果你想把这个函数声明为 class func

class func moveText(){ 
    // Your Code
}
           

则本文最开始的代码中的 target 应该由 self 换为 MyClass.self (假设这段代码写在名为 MyClass 的类中)

self.timer = NSTimer.scheduledTimerWithTimeInterval(timeInterval, target: MyClass.self,
        selector: "moveText", userInfo: nil, repeats: true)
           

最后, 以上关键字是可以叠加使用的,如果你愿意,可以写成下面这样:

@objc private class func moveText(){ 
    // Your Code
}
           

我在 StackOverFlow 中的回答,及其它相关:

http://stackoverflow.com/questions/30027322/how-to-call-private-class-function-from-selector-in-swift/34291667#34291667 http://stackoverflow.com/questions/28184350/4-questions-about-selectors-in-swift

如有谬误,欢迎在评论中指出。