美文网首页
Swift实现代理,通知,闭包传值

Swift实现代理,通知,闭包传值

作者: iOScoderZZJ | 来源:发表于2016-07-21 11:25 被阅读958次

    前言

    记得刚刚学习OC的时候被各个界面间的传值绕的云里雾里的,通知\代理\block的使用,更是有很多不理解的地方,在学习Swift的道路中,发现这些还是必须要学会的,这篇文章算是给自己的一个小总结,文章中讲的例子已经上传至github <a href="https://github.com/iOScoderZZJ/ZZJSwift_noti_delegate_block">demo下载</a>,不足的地方多多批评指正_,效果图如下

    aaaa.gif

    1.区别

    <a href="http://www.jianshu.com/p/33753c326232">原文链接</a>,著作权归作者所有,侵权删
    第一就是代理,这也是很常用的方式,特点是一对一的形式,而且逻辑结构非常清晰。实现起来较为简单:写协议 ,设置代理这个属性, 最好在你想通知代理做事情的方法中调用即可。当然这里面有一些细节,包括 ①协议定义时,请用关键字@required,和@optional来明确代理是否必须实现某些方法 ②代理的类型需用id类型,并写明要遵守的协议 ③就是在调用代理方法的时候需要判断代理是否实现该方法。

    第二就是通知,通知的优点很明显,他是一对多的形式,而且可以在任意对象之间传递,不需要二者有联系,当然他的实现和代理相比较稍微绕一点,注册,发通知,收通知。这里面的注意点就是 ①对于系统没有定义的事件监听时需要自己发通知,这是你就需要定义一个key,字符串类型,这也是通知的一个弊端,你需要拷贝到收通知的对象,避免写错一个字母而无法收通知的尴尬 ②就是注册的通知中心需要手动移除,不然除了性能的问题还会有其他的问题出现,比如说一个控制器消失了之后还有因为某些事件而发出通知,造成不想要的结果。

    第三就是block了,这是苹果后来才加入的,也是目前开发比较常用的一种方式,功能比较强大,但是在理解和使用上可能需要一段时间摸索和熟悉。他的最大特点就是回调,而且回调时可以传入参数,最重要的是,无论在哪调用,block的执行都会回到block创建的地方执行,而非调用的地方。而block本身可以封装一段代码,一段代码你懂的,很多人在初学时会被搞晕,甚至在block的声明上就纠结,其实很正常,多用就好。

    这是别人总结的OC中三者的区别,在Swift的开发中,block被闭包代替(和block极其相似),我认为两者就是替换与被替换的关系,三者在Swift中的写法和一些注意事项会在下文继续阐述

    2.1代理

    如下面代码,我们定义了一个协议方法, 其结构就是protocol关键字+协议的名称 + : + NSObjectProtocol {},注意Swift中的自定义协议其本身必须遵循NSObjectProtocol协议

    protocol TextViewControllerDelegate: NSObjectProtocol{
        // 点击delegate按钮回调
        func delegateBtnWillClick(message : String)
    }
    

    并且在OC中,我们一般会定义一个属性来保存代理如@property (nonatomic,weak)id< TextViewControllerDelegate > delegate;但是在Swift中我们依然要这样做,只不过写法变成了weak var delegate : TextViewControllerDelegate?,注意一定要加上weak关键字,避免循环引用问题.这是我的demo中的代码,回调一段字符串

    //MARK: - 点击代理传值按钮
        func delegateBtnClick()
        {
            print("代理传值按钮被点击")
            let str = "代理传值按钮被点击,把上个界面的值传了过来"
            delegate?.delegateBtnWillClick(str)
            self.navigationController?.popViewControllerAnimated(true)
        }
    

    然后就在别的控制器,设置那个控制器为该控制器的代理,去实现协议方法就可以了,具体代码见demo

    2.2通知

    我在代码中实现了在一个控制器中,点击按钮发送通知,另一个控制器监听通知,首先发送通知写法如下(带参数)

    //发送通知
    let dict = ["action":"通知传值按钮被点击,把上个界面的值传了过来"]
    NSNotificationCenter.defaultCenter().postNotificationName(notificationBtnWillClick, object: self, userInfo: dict)
    

    在另一个控制器中监听通知

    //监听通知
     NSNotificationCenter.defaultCenter().addObserver(self, selector: "receiveNoti:", name: notificationBtnWillClick, object: nil)
    //实现receiveNoti:通知方法
    func receiveNoti(notification : NSNotification)
        {
            let userInfo = notification.userInfo
            print(userInfo!["action"])
        }
    

    这样我们把上个界面的字典传递过来了

    2.3闭包

    闭包和OC中的block非常相似, OC中的block类似于匿名函数,闭包是用来定义函数,作用:block是用于保存一段代码,在需要的时候执行,闭包也是保存一段代码,在需要的时候执行

    闭包的基本格式:
            in的含义用于区分形参返回值和执行代码
            {
                (形参列表) -> (返回值)
                in
                需要执行的代码
            }
    

    大家也许不太理解,我给大家举个例子,在OC中,GCD大家肯定都用过

    //该函数后面跟着一个block 
    dispatch_async(dispatch_get_main_queue(), ^{
            
     });
    //在Swift中我们再来敲下这个函数,发现Swift中该函数后面跟的就是闭包
    dispatch_async(dispatch_get_main_queue()) { () -> Void in
                
     }
    

    所以以后开发中闭包形式忘了,一言不合敲波dispatch_...就能帮你回想起不少东西
    定义一个闭包,参数为String类型,没有返回值typealias blockBtnClickSendValue = (message:String)->Void其次用属性保存一波var myblock : blockBtnClickSendValue?在恰当出发的地方

             if myblock != nil
            {
                myblock!(message: "闭包传值按钮被点击,把上个界面的值传了过来")
            }
    

    然后在另一个控制器操作就可以了

    textVC.myblock =  {(message:String) -> Void in
                print("\(message)")
    }
    

    3.结束

    这里带大家简简单单的入了个门,自己有的地方也是有点懵懵哒,多敲多看没有什么捷径可以走,有的时候真的感觉自己的脑子不共用,嘿嘿,慢慢来
    我是一个iOS开发的小菜鸡,希望有天可以成为一只雄鹰
    旅途很远,还需修行

    相关文章

      网友评论

          本文标题:Swift实现代理,通知,闭包传值

          本文链接:https://www.haomeiwen.com/subject/mfjbjttx.html