美文网首页带我飞2Swiftswift-问题栈
[Swift2.0系列]Protocol Extensions

[Swift2.0系列]Protocol Extensions

作者: NinthDay | 来源:发表于2015-09-26 15:54 被阅读1800次

    1.Class 和 Struct

    要知道ObjC,Swift都是面向对象编程,编程应用中:通常先定义一个基类(base Class),紧接着为该类添加一些基础属性以及方法,这便是父类(super class);倘若现在要声明一个继承于它的子类,通常是这么声明的class ChildClass:SuperClass{},这时候该子类就拥有了父类的所有特性(包括属性和方法),子类还可以通过override来重写父类的计算属性以及方法,但是却无法重写父类的存储属性,这点值得引起注意。为了直观表达,简单写个小demo:

    // 父类
    class SuperClass{
        // 包含一个name属性
        var storedVar:String = "storedVar"
        var computedVar:String {
            return "computedVar"
        }
        // 打印名字
        func descriptionClass(){
            print("storedVar is \(storedVar)")
            print("computedVar is \(computedVar)")
        }
    }
    // 子类2
    class FirstChildClass:SuperClass{
        // 可以什么都不写
    }
    // 子类2
    class SecondChildClass:SuperClass{
        // 重写父类属性中的名字
        override var computedVar:String{
            return "修改后的computedVar"
        }
        override func descriptionClass(){
            print("cc storedVar is \(storedVar)")
            print("cc computedVar is \(computedVar)")
        }
    }
    var fc = FirstChildClass()
    var sc = SecondChildClass()
    fc.descriptionClass()   
    sc.descriptionClass()   
    

    Class暂时告一段落,我迫不及待想要向你推销Struct了!

    • A:重写上述代码,用结构体??
    • B:不不!结构体是值类型,且不支持继承。
    • A:难道用协议?
    • B:Bingo!
    • A:倒是一个不错的注意,指定一个协议,然后让两个子类遵循这个协议即可,不过....
    • B:看来你也知晓弊端了,遵循协议后,必须各自实现协议中的方法,由于结构体又是值类型,那么相当于方法与结构体之间是一对一,而非Class中子类只要继承父类,共同指向父类中的方法。

    这段对话中协议的应用代码如下:

    protocol ClassProtocol{
        var name:String{get set}
        func description()
    }
    
    struct FirstClass:ClassProtocol{
        var name = "firstClass"
        func description() {
            print("First class")
        }
    }
    
    struct SecondClass:ClassProtocol {
        var name = "firstClass"
        func description() {
            print("Second class")
        }
    }
    

    显然这并不是我们想要的结果,和类中继承大相径庭。

    不过Swift2.0语法的更新为我们带来了协议上的改动,增加了Extension

    或许你还不理解,首先Extension译为扩展、拓展,顾名思义就是对已声明的协议补充额外的默认行为。这里我不得不友情提示,Extension中哪些内容是不被允许的。

    1. 有童鞋声明了一个协议,却遗漏了某个协议方法,这时候想要通过extension来补充,例如extension ClassProtocol{func extraMethod()}会提示报错"expected '{' in body of function declare",先前说了扩展中需要添加额外的默认实现,而非协议中只要求方法声明即可满足。
    2. 存储属性。例如extension ClassProtocol{ var name2:String = "hello"}是不被允许的。

    使用面向协议编程

    不如来个角色扮演游戏来讲解:游戏中,有敌人Enemy,我们规定敌人要有名字以及攻击手段,显然不同敌人都各有不同,但是有一点我们规定是一样的:死的时候大喊一声“我挂了”(调皮了)。为此先制定一个协议。

    protocol Enemy{
        var name:String{get set}    //名字
        func attack()   //攻击手段
    }
    extension Enemy{
        func deadBehavior(){
            print("挂了")
        }
    }
    // 小罗罗
    struct SimpleEnemy:Enemy{
        var name = "小啰啰"
        func attack(){
            print("用刀砍")
        }
    }
    // Boss
    struct Boss:Enemy{
        var name = "boss"
        func attack(){
            print("用枪打")
        }
    }
    
    var s = SimpleEnemy()
    var b = Boss()
    s.deadBehavior()
    b.deadBehavior()    //可以看到通过协议扩展它们拥有共同的方法了!!!
    

    倘若终极Boss想要特立独行点,死时候的喊的霸气点,我们还可以重写默认的协议方法呢!

    struct SuperBoss:Enemy{
        var name = "终极boss"
        func attack(){
            print("无所不能")
        }
        func deadBehavior(){
            print("我还会再回来的")
        }
    
    }
    var sb = SuperBoss()
    sb.deadBehavior()
    

    可以看到重写了扩展中的默认方法,最终会调用SuperBoss()中重写的deadBehavior()方法。

    为了让游戏好玩,我设定了终极Boss死后时会蜕变成另外一个Emeny

    var sb = SuperBoss()
    sb.deadBehavior()
    var weakSb = sb as Enemy
    weakSb.deadBehavior()// 输出挂了
    

    可以看到终极Boss变成Enemy之后不再拥有原先的“我还会再回来的”死亡行为。

    10.16日更新添加内容
    这是采用面向协议实现的字符校验第三方库,可以作为学习或者项目使用。地址PTStringValidator Framework

    总结:本文难度不大,旨在为大家讲解Protocol Extension的基础以及实际应用,倘若觉得不错,请订阅我以及点击喜欢。

    开设了一个swift交流群:392436022 欢迎加入。

    相关文章

      网友评论

      • 巴图鲁:膜拜
        NinthDay:@巴图鲁 谢谢
      • 逸小山:博主写的不错,赞一个!话说我加群了,然后发现群里应该就博主一人,然后发现博主竟然是九江的,我是新余的,老乡你好 :wink:
      • 瓶子里的风:请问weakSb 的attack 为什么还是无所不能?
        NinthDay:@瓶子里的风 var weakSb = sb as Enemy 这条语句后 weakSb的类型其实就是Enemy,weakSb,那么优先会在Enemy协议中找有没有默认实现,如果有就调用。那么attack也是协议中定义的一个方法。但是没有默认实现,所以回去对应的实例中找吧。我猜测是 swift runtime 干的事情。附上raywenderlich的一篇问答文章:https://www.raywenderlich.com/110982/swift-interview-questions-answers。里面有相关面试题,不过是作者自己出的。。。。。

      本文标题:[Swift2.0系列]Protocol Extensions

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