实现目标与动作关联使用UIControl类addTarget(_:action:forControlEvents:)方法,示例代码如下:
1
2
|
button.addTarget(self,action:
"onClick:"
,
forControlEvents: UIControlEvents.TouchUpInside)
|
其中的action参数"onClick:"事实上就是选择器(Selector)。
问题提出
任何能够将方法调用的绑定推迟到运行期,在编译时方法调用者不需要知道要调用的方法是什么,这个可以降低调用者与被调用者之间的耦合度,这样就语言就很灵活。在C语言在提供一种函数指针技术,Objective-C和Swift语言都提供选择器(Selector)类型,它是C语言函数指针的面向对象替代技术。
选择器在Cocoa和Cocoa Touch中的目标动作、通知和委托等模式中方法的调用实现的关键。
解决方案
Objective-C中选择器是SEL数据类型,使用@selector()语句调用,调用onClick:方法的Objective-C示例代码如下:
1
2
3
|
SEL selector =
@selector
(onClick:);
[button addTarget:selfaction: selector
forControlEvents:UIControlEventTouchUpInside];
|
Swift中虽然没有提供SEL数据类型,而是提供了Selector结构体,通过方法名字符串构建Selector实例,示例代码如下:
1
2
|
button.addTarget(self,action: Selector(
"onClick:"
),
forControlEvents: UIControlEvents.TouchUpInside)
|
通过选择器调用方法,关键是方法名字,它有一定规律的。穷其根本是源自于Objective-C多重参数方法命名规律。方法名的冒号暗示了方法名应该具有几个参数,下面我们看几个示例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
//选择器为"onClick:"
func onClick(sender: AnyObject) {
NSLog(
"onClick:"
)
}
//选择器为"onClick:forEvent:"
func onClick(sender: AnyObject, forEventevent: UIEvent) {
NSLog(
"onClick:forEvent:"
)
}
//选择器为"onClickWithExtSender:forEvent:"
func onClick(extSender sender: AnyObject,forEvent event: UIEvent) {
NSLog(
"onClickWithExtSender:forEvent:"
)
}
|
出于数据封装的需要,我们会在方法前面加private,使其变为私有方法,代码如下。
1
2
3
|
private
func onClick(sender: AnyObject) {
NSLog(
"onClick:"
)
}
|
但是这样方法在调用时候会出现如下错误:
1
|
unrecognized selector sentto instance
0x7f7f81499b10
'
|
这个错误的意思是没有找到选择器所指定的方法,也就是没有找到onClick:方法。正确的做法是在方法前面添加@objc属性注释,这说明选择器是在objc runtime运行环境下调用的。
1
2
3
4
|
//选择器为"onClick:"
@objc
private
func onClick(sender:AnyObject) {
NSLog(
"onClick:"
)
}
|
本文转自 tony关东升 51CTO博客,原文链接:http://blog.51cto.com/tonyguan/1748676,如需转载请自行联系原作者