由于種種原因鳞尔,簡(jiǎn)書等第三方平臺(tái)博客不再保證能夠同步更新彼念,歡迎移步 GitHub:https://github.com/kingcos/Perspective/狞尔。謝謝斯议!
Selectors in Swift
- Info:
- Swift 3.0
- Xcode 8.2.1
- macOS 10.12.4 beta (16E144f)
前言
今天是大年初四(捂臉:提筆的時(shí)候是初一),總算過農(nóng)歷新年了乞旦,總算可以歇一歇了贼穆。越來越感慨時(shí)間過得飛快,計(jì)劃總是趕不上變化兰粉。寒假倒計(jì)時(shí) 20 天故痊,卻有很多事都還沒有完成。玖姑。
常用純代碼來開發(fā)的同學(xué)都應(yīng)該比較熟悉這個(gè)方法:
func addTarget(_ target: Any?, action: Selector, for controlEvents: UIControlEvents)
Selector 源自 Objective-C愕秫,例如 SEL 類型,以及 @selector()
方法選擇器客峭。Swift 中也兼容了這個(gè)概念豫领,不過隨著 Swift 的迭代,Selector 的一些寫法也出現(xiàn)了很大的變化舔琅。比較遺憾的是,官方文檔對(duì)于 Selector 沒有介紹洲劣。
因此只能自己總結(jié)一下 Swift 3.0 中的 Selector备蚓,便有利于自己理解,也便于以后的參考囱稽。注:以下 Demo 中的 cyanButton 是用 StoryBoard 拖拽的郊尝。
Selector 類型
Swift 中的 Selector 類型其實(shí)就是 Objective-C 中的 SEL 類型。在 Swift 中战惊,Selector 的本質(zhì)是結(jié)構(gòu)體流昏。常用的構(gòu)造 Selector 類型變量的方法有以下幾種:
public init(_ str: String)
類似 Objective-C 中的 NSSelectorFromString
,Swift 中的 Selector 也可以使用字符串來構(gòu)造:
@IBOutlet weak var cyanButton: UIButton!
override func viewDidLoad() {
super.viewDidLoad()
cyanButton.addTarget(self,
action: Selector("cyanButtonClick"),
for: .touchUpInside)
}
func cyanButtonClick() {
print(#function)
}
#selector()
通過字符串構(gòu)造 Selector 變量是一種方法吞获,但是當(dāng)在上例中 Xcode 會(huì)提示這樣的警告:「Use '#selector' instead of explicitly constructing a 'Selector'」况凉。即使用 #selector()
代替字符串明確構(gòu)造 Selector。
@IBOutlet weak var cyanButton: UIButton!
override func viewDidLoad() {
super.viewDidLoad()
cyanButton.addTarget(self,
action: #selector(ViewController.cyanButtonClick),
for: .touchUpInside)
}
func cyanButtonClick() {
print(#function)
}
#selector()
的好處是不再需要使用字符串來構(gòu)造各拷。因?yàn)楫?dāng)使用字符串構(gòu)造時(shí)刁绒,若傳入的字符串沒有對(duì)應(yīng)的方法名,那么程序在執(zhí)行時(shí)就會(huì)直接崩潰:「unrecognized selector sent to instance」烤黍。
若當(dāng)前作用域構(gòu)造 Selector 的方法名唯一時(shí)知市,可以直接使用方法名傻盟,而省略作用域。
cyanButton.addTarget(self,
action: #selector(cyanButtonClick),
for: .touchUpInside)
若是 Swift 中的私有方法嫂丙,則必須賦予其 Objective-C 的 runtime(運(yùn)行時(shí))娘赴。即在方法名前加上 @objc
:
@IBOutlet weak var cyanButton: UIButton!
@IBOutlet weak var anotherCyanButton: UIButton!
override func viewDidLoad() {
super.viewDidLoad()
cyanButton.addTarget(self,
action: #selector(ViewController.cyanButtonClick(_:)),
for: .touchUpInside)
// 當(dāng)前作用域 cyanButtonClick 存在沖突,不能直接使用方法名
//「Ambiguous use of 'cyanButtonClick'」
// anotherCyanButton.addTarget(self,
action: #selector(cyanButtonClick),
for: .touchUpInside)
}
// 無參方法
func cyanButtonClick() {
print(#function)
}
// 有參私有方法
@objc private func cyanButtonClick(_ button: UIButton) {
let btnLabel = button.titleLabel?.text ?? "nil"
print(btnLabel)
print(#function)
}
當(dāng)遇到上述存在歧義的相同方法名時(shí)跟啤,也可以使用強(qiáng)制類型轉(zhuǎn)換來解決:
@IBOutlet weak var cyanButton: UIButton!
@IBOutlet weak var anotherCyanButton: UIButton!
override func viewDidLoad() {
super.viewDidLoad()
let methodA = #selector(cyanButtonClick as () -> ())
let methodB = #selector(cyanButtonClick as (UIButton) -> ())
cyanButton.addTarget(self,
action: methodA,
for: .touchUpInside)
anotherCyanButton.addTarget(self,
action: methodB,
for: .touchUpInside)
}
func cyanButtonClick() {
print(#function)
}
@objc private func cyanButtonClick(_ button: UIButton) {
let btnLabel = button.titleLabel?.text ?? "nil"
print(btnLabel)
print(#function)
}
-
#selector()
&Seletcor("")
通過上面的 Demo筝闹,也可以看出 #selector()
更加安全、清晰腥光,但是 Seletcor("")
并不是一無是處关顷。當(dāng)我們需要調(diào)用標(biāo)準(zhǔn)庫中的私有方法時(shí),只能通過字符串來構(gòu)造武福。
為了方便測(cè)試议双,此處自定義了一個(gè) CustomViewController
。其中帶有私有方法:@objc private func privateFunc()
以及 func defaultFunc()
捉片。此處使用的 ViewController
繼承自 CustomViewController
:
CustomViewController.swift
class CustomViewController: UIViewController {
@objc private func privateFunc() {
print(#function)
}
func defaultFunc() {
print(#function)
}
}
ViewController.swift
class ViewController: CustomViewController {
@IBOutlet weak var cyanButton: UIButton!
@IBOutlet weak var anotherCyanButton: UIButton!
override func viewDidLoad() {
super.viewDidLoad()
cyanButton.addTarget(self,
action: #selector(defaultFunc),
for: .touchUpInside)
anotherCyanButton.addTarget(self,
action: Selector("privateFunc"),
for: .touchUpInside)
}
}
因?yàn)楦割惖乃接蟹椒▽?duì)子類來說是不可見的平痰,直接使用 #selector()
無法通過編譯,但這個(gè)方法確實(shí)存在伍纫,所以這里只能使用字符串來構(gòu)造 Selector宗雇。
當(dāng)然這里 Xcode 會(huì)提示警告,但仍然可以編譯通過并運(yùn)行莹规,所以這并不是官方提倡的行為赔蒲。這是我在將系統(tǒng)邊緣返回改寫全屏返回時(shí),發(fā)現(xiàn)私有的 handleNavigationTransition:
方法不能通過 #selector()
良漱,因此使用了字符串代替舞虱。
Syntax Sugar
配合 Swift 的 Extension,可以使用其管理當(dāng)前控制器的所有 Selector:
import UIKit
fileprivate extension Selector {
static let redButtonClick = #selector(ViewController.redButtonClick(_:))
static let cyanButtonClick = #selector(ViewController.cyanButtonClick)
}
class ViewController: CustomViewController {
@IBOutlet weak var cyanButton: UIButton!
@IBOutlet weak var redButton: UIButton!
override func viewDidLoad() {
super.viewDidLoad()
cyanButton.addTarget(self,
action: .cyanButtonClick,
for: .touchUpInside)
redButton.addTarget(self,
action: .redButtonClick,
for: .touchUpInside)
}
func cyanButtonClick() {
print(#function)
}
func redButtonClick(_ button: UIButton) {
let btnLabel = button.titleLabel?.text ?? "nil"
print(btnLabel)
print(#function)
}
}
getter & setter
Swift 3.0 中加入了 Selector 引用變量(不可為常量)的 getter 和 setter 方法:
class Person: NSObject {
dynamic var firstName: String
dynamic let lastName: String
dynamic var fullName: String {
return "\(firstName) \(lastName)"
}
init(firstName: String, lastName: String) {
self.firstName = firstName
self.lastName = lastName
}
}
fileprivate extension Selector {
static let firstNameGetter = #selector(getter: Person.firstName)
static let firstNameSetter = #selector(setter: Person.firstName)
}