美文网首页程序员
Swift中的继承(Inheritance)

Swift中的继承(Inheritance)

作者: 扑腾的蛾子 | 来源:发表于2022-07-04 16:59 被阅读0次

    继承(Inheritance)

    1、值类型(枚举、结构体)不支持继承,只有类支持继承

    2、没有父类的类,称为:基类

        Swift并没有像OC、Java那样的规定:任何类最终都要继承自某个基类。
    

    3、子类可以重写父类的下标、方法、属性,重写必须加上override关键字。

    内存结构

    class Animal {
        var age = 0
    }
     
    class Dog : Animal {
        var weight = 0
    }
     
    class ErHa : Dog {
        var iq = 0
    }
    
    let a = Animal()
    a.age = 10
    

    看一下a需要多少内存,a是堆空间的,所以必然是16的倍数,最前面有8个字节拿来放类型信息,第二个8个字节放引用计数相关的东西,再往后8个字节才是放age,总共用到的是24个字节,但是需要保证是16的倍数,所以是32个字节。

    Dog因为有继承,所以等价于

    class Animal {
        var age = 0
    }
     
    class Dog : Animal {
        var weight = 0
    }
     
    class Dog {
        var age = 0
        var weight = 0
    }
     
    class ErHa : Dog {
        var iq = 0
    }
    
    let d = Dog()
    d.age = 10
    d.weight = 20
    

    d对象里面有两个属性,age和weight,各占8个字节,并且一般来说父类的属性内存靠前,d对象也占用32个字节,第一块是类型相关的8个字节,第二块是引用计数相关的8个字节,第三块是存储age的8个字节,第四块是存储weight的8个字节。

    同样的一个ErHa对象要有24个字节存储age、weight、iq,另外还有前面的16个字节,所以是40个字节,但是要保证是16的倍数,所以就是48。

    重写实例方法、下标

    class Animal {
        func speak() {
            print("Animal speak")
        }
        
        subscript(index: Int) -> Int {
            return index
        }
    }
     
    class Cat : Animal {
        override func speak() {
            super.speak()
            print("Cat speak")
        }
        
        override subscript(index: Int) -> Int {
            return super[index] + 1
        }
    }
     
    var anim: Animal
    anim = Animal()
    //Animal speak
    anim.speak()
    //6
    print(anim[6])
     
    anim = Cat()
    //Animal speak
    //Cat speak
    anim.speak()
    // 7
    print(anim[6])
    

    重写类型方法、下标

    1、被class修饰的类型方法、下标,允许被子类重写

    2、被static修饰的类型方法、下标,不允许被子类重写

    class Animal {
        class func speak() {
            print("Animal speak")
        }
        
        class subscript(index: Int) -> Int {
            return index
        }
    }
     
    class Cat : Animal {
        override class func speak() {
            super.speak()
            print("Cat speak")
        }
        
        override class subscript(index: Int) -> Int {
            return super[index] + 1
        }
    }
    
    static修饰的类型方法、下标重写报错
    子类重写可以用static修饰,只不过不能再继续被重写了

    重写属性

    1、子类可以将父类的属性(存储、计算)重写为计算属性

    2、子类不可以将父类属性重写为存储属性

    3、只能重写var属性,不能重写let属性

    4、重写时,属性名、类型要一致

    5、子类重写后的属性权限,不能小于父类属性的权限

        如果父类属性是只读的,那么子类重写后的属性可以是只读的,也可以是可读写的。
    
        如果父类属性是可读写的,那么子类重写后的属性也必须是可读写的。
    

    重写实例属性

    class Circle {
        var radius: Int = 0
        var diameter: Int {
            set {
                print("Circle setDiameter")
                radius = newValue / 2
            }
            get {
                print("Circle getDiameter")
                return radius * 2
            }
        }
    }
     
    class SubCircle: Circle {
        override var radius: Int {
            set {
                print("SubCircle setRadius")
                super.radius = newValue > 0 ? newValue : 0
            }
            get {
                print("SubCircle getRadius")
                return super.radius
            }
        }
        override var diameter: Int {
            set {
                print("SubCircle setDiameter")
                super.diameter = newValue > 0 ? newValue : 0
            }
            get {
                print("SubCircle getDiameter")
                return super.diameter
            }
        }
    }
    
    var circle = SubCircle()
    circle.radius = 6
    //SubCircle setRadius
     
    print(circle.diameter)
    //SubCircle getDiameter
    //Circle getDiameter
    //SubCircle getRadius
    //12
     
    circle.diameter = 20
    //SubCircle setDiameter
    //Circle setDiameter
    //SubCircle setRadius
     
    print(circle.radius)
    //SubCircle getRadius
    //10
    

    重写类型属性

    1、被class修饰的计算类型属性,可以被子类重写

        存储类型属性只能用static来修饰。
    

    2、被static修饰的类型属性(存储、计算),不可以被子类重写

    属性观察器

    1、可以在子类中为父类属性(除了只读计算属性、let属性)增加属性观察器

    class Circle {
        var radius: Int = 1
    }
     
    class SubCircle: Circle {
        override var radius: Int {
            willSet {
                print("SubCircle willSetRadius", newValue)
            }
            didSet {
                print("SubCircle didSetRadius", oldValue, radius)
            }
        }
    }
     
    var circle = SubCircle()
    circle.radius = 10
    //SubCircle willSetRadius 10
    //SubCircle didSetRadius 1 10
    

    2、父类和子类中都有属性观察器

    
    class Circle {
        var radius: Int = 1 {
            willSet {
                print("Circle willSetRadius", newValue)
            }
            didSet {
                print("Circle didSetRadius", oldValue, radius)
            }
        }
    }
     
    class SubCircle: Circle {
        override var radius: Int {
            willSet {
                print("SubCircle willSetRadius", newValue)
            }
            didSet {
                print("SubCircle didSetRadius", oldValue, radius)
            }
        }
    }
     
    var circle = SubCircle()
    circle.radius = 10
    //SubCircle willSetRadius 10
    //Circle willSetRadius 10
    //Circle didSetRadius 1 10
    //SubCircle didSetRadius 1 10
    

    3、子类是可以给父类中的计算属性增加属性观察器的。

    class Circle {
        class var radius: Int {
            set {
                print("Circle setRadius", newValue)
            }
            get {
                print("Circle getRadius")
                return 20
            }
        }
    }
     
    class SubCircle: Circle {
        override static var radius: Int {
            willSet {
                print("SubCircle willSetRadius", newValue)
            }
            didSet {
                print("SubCircle didSetRadius", oldValue, radius)
            }
        }
    }
     
    SubCircle.radius = 10
    //        Circle getRadius (oldValue)
    //        SubCircle willSetRadius 10
    //        Circle setRadius 10
    //        Circle getRadius (radius)
    //        SubCircle didSetRadius 20 20
    

    final

    1、被final修饰的方法、下标、属性,禁止被重写

    2、被final修饰的类,禁止被继承

    相关文章

      网友评论

        本文标题:Swift中的继承(Inheritance)

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