原创文章,欢迎转载。转载请注明:关东升的博客

实现目标与动作关联使用UIControl类addTarget(_:action:forControlEvents:)方法,示例代码如下:


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示例代码如下:


SEL selector = @selector(onClick:);
[button addTarget:self action: selector
      forControlEvents: UIControlEventTouchUpInside];


Swift中虽然没有提供SEL数据类型,而是提供了Selector结构体,通过方法名字符串构建Selector实例,示例代码如下:


button.addTarget(self, action: Selector("onClick:"),
      forControlEvents: UIControlEvents.TouchUpInside)


通过选择器调用方法,关键是方法名字,它有一定规律的。穷其根本是源自于Objective-C多重参数方法命名规律。方法名的冒号暗示了方法名应该具有几个参数,下面我们看几个示例:
  

  //选择器为"onClick:"
    func onClick(sender: AnyObject) {
        NSLog("onClick:")
    }
   
    //选择器为"onClick:forEvent:"
    func onClick(sender: AnyObject, forEvent event: UIEvent) { 
        NSLog("onClick:forEvent:")
    }
   
    //选择器为"onClickWithExtSender:forEvent:"
    func onClick(extSender sender: AnyObject, forEvent event: UIEvent) {
        NSLog("onClickWithExtSender:forEvent:")
    }

 

出于数据封装的需要,我们会在方法前面加private,使其变为私有方法,代码如下。


    private func onClick(sender: AnyObject) {
        NSLog("onClick:")
    }


但是这样方法在调用时候会出现如下错误:
unrecognized selector sent to instance 0x7f7f81499b10'
这个错误的意思是没有找到选择器所指定的方法,也就是没有找到onClick:方法。正确的做法是在方法前面添加@objc属性注释,这说明选择器是在objc runtime运行环境下调用的。
   

//选择器为"onClick:"
    @objc private func onClick(sender: AnyObject) {
        NSLog("onClick:")
    }


欢迎关注关东升新浪微博@tony_关东升。
关注智捷课堂微信公共平台,了解最新技术文章、图书、教程信息
  
更多精品iOS、Cocos、移动设计课程请关注智捷课堂官方网站:http://www.zhijieketang.com
智捷课堂论坛网站:http://51work6.com/forum.php