13.Swift-继承与重写


一个类可以继承另一个类的方法、属性和其他特性。在Swift中,继承是类区分与其他类型的一个基本特性。

在Swift中,子类可以调用和访问父类的方法、属性和附属脚本,并且可以通过重写这些方法、属性和附属脚本来优化和修改它们的行为。Swift会检查你的重写定义在父类中是否有匹配的定义来确保你的重写行为是正确的。

类也可以为从父类中继承来的属性添加属性监视器,当属性值改变时,类就会被通知到。可以为任何属性添加属性监视器,无论它原本被定义为存储属性还是计算属性。


一、定义一个基类

任何不继承与其他类的类都称为基类。

注意:Swift 中的类并不是从一个通用的基类继承而来。如果你不为你定义的类指定一个超类的话,这个类就自动成为基类。

下面定义了一个叫Vehicle的基类,这个基类定义了一个存储属性currentSpeed,它的默认值是0.0。currentSpeed被另一个只读字符串属性description用来生成一个对vehicle的描述。

Vehicle也定义了一个makeNoise方法,这个方法在基类的实例中并没有做任何操作,但是将在Vehicle的子类的实例中做具体操作。

class Vehicle{

    var currentSpeed = 0.0

    var description:String{

        return "traveling at \(currentSpeed) miles per hour"

    }

    func makeNoise(){

        //do nothing - an arbitrary vehicle doesn't necessary make a noise

    }

}


let someVehicle = Vehicle()

println("Vehicle:\(someVehicle.description)")//Vehicle:traveling at 0.0 miles per hour

Vehicle类定义了车辆的共同特性,但这个类本身并没太大用处。为了使它更为实用,你需要进一步细化它来描述更具体的车辆。


二、生成子类

子类生成(Subclassing)指的是在一个已有类的基础上创建一个新的类。子类继承超类的特性,并且可以优化或改变它。你还可以为子类添加新的特性。

 

为了指明某个类的超类,将超类名写在子类名的后面,用冒号分隔:

class SomeSubclass:Vehicle{

    //subclass definition goes here

}

下面定义一个Vehicle的子类Bicycle:

class Bicycle:Vehicle{

    var hasBasket = false

}

Bicycle类自动获得了它的父类Vehicle的所有特性,比如currentSpeed和description属性和makeNoise方法。Bicycle还定义了新的hasBasket属性,默认值为false。默认新建一个Bicycle实例,它的hasBasket属性为false,你可以为特定的Bicycle实例的hasBasket属性赋值为true。

let bicycle = Bicycle()

bicycle.hasBasket = true

你可以修改继承来的currentSpeed属性,还可以访问继承来的description属性:

bicycle.currentSpeed = 15.0

println("Bicycle:\(bicycle.description)")//Bicycle:traveling at 15.0 miles per hour


子类本身又可以被继承,下面创建了一个Bicycle的子类Tandem:

class Tandem:Bicycle{

    var currentNumberOfPassengers = 0

}

Tandem继承了Bicycle的所有属性和方法,也继承了Vehicle的所有属性和方法。Tandem也新增了一个默认值为0的存储属性currentNumberOfPassengers。如果你新建一个Tandem的实例,你可以访问它的新增属性和继承来的属性,还可以获取它从Vehicle继承来的description属性:

let tandem = Tandem()

tandem.hasBasket = true

tandem.currentNumberOfPassengers = 2

tandem.currentSpeed = 22.0

println("Tandem:\(tandem.description)")//Tandem:traveling at 22.0 miles per hour


三、重写

子类可以为继承来的实例方法、类方法、实例属性或者附属脚本提供自己定制的实现,这种行为成为重写。

如果要重写某个特性,你需要在重写定义的前面加上override关键字。这么做,你就表明了你是想提供一个重写版本,而非错误地提供了一个相同的定义。意外的重写行为可能会导致不可预知的错误,任何缺少override关键字的重写都会在编译时被诊断为错误。

 

override关键字会提醒 Swift 编译器去检查该类的超类(或其中一个父类)是否有匹配重写版本的声明。这个检查可以确保你的重写定义是正确的。


1、访问父类的方法、属性和附属脚本

当你在子类中重写超类的方法,属性或附属脚本时,有时在你的重写版本中使用已经存在的超类实现会大有裨益。比如,你可以优化已有实现的行为,或在一个继承来的变量中存储一个修改过的值。

在合适的地方,你可以通过使用super前缀来访问超类版本的方法,属性或附属脚本:


在方法someMethod的重写实现中,可以通过super.someMethod()来调用超类版本的someMethod方法。

在属性someProperty的 getter 或 setter 的重写实现中,可以通过super.someProperty来访问超类版本的someProperty属性。

在附属脚本的重写实现中,可以通过super[someIndex]来访问超类版本中的相同附属脚本。


2、重写方法

在子类中,你可以重写继承来的实例方法或类方法,提供一个定制或替代的方法实现。

下面定义了一个Vehicle的子类Train,它重写了从Vehicle继承来的makeNoise方法:

class Train:Vehicle{

    override func makeNoise() {

        println("Choo Choo")

    }

}

let train = Train()

train.makeNoise()//Choo Choo


3、重写属性

你可以重写继承来的实例属性或类属性,提供自己定制的getter和setter,或添加属性观察器使重写的属性观察属性值什么时候发生改变。

class Train:Vehicle{

    override func makeNoise() {

        println("Choo Choo")

    }

    

    override var currentSpeed:Double { //需要标明类型

        get{ //有set必须提供get

            return 10.0

        }

        set{

        

        }

    }

}


4、重写属性的getters和setters

你可以提供定制的 getter(或 setter)来重写任意继承来的属性,无论继承来的属性是存储型的还是计算型的属性。子类并不知道继承来的属性是存储型的还是计算型的,它只知道继承来的属性会有一个名字和类型。你在重写一个属性时,必需将它的名字和类型都写出来。这样才能使编译器去检查你重写的属性是与超类中同名同类型的属性相匹配的。

你可以将一个继承来的只读属性重写为一个读写属性,只需要你在重写版本的属性里提供 getter 和 setter 即可。但是,你不可以将一个继承来的读写属性重写为一个只读属性。

注意:如果你在重写属性中提供了 setter,那么你也一定要提供 getter。如果你不想在重写版本中的 getter 里修改继承来的属性值,你可以直接返回super.someProperty来返回继承来的值。


下面定义一个Vehicle的子类Car,它定义了一个存储属性gear,默认值是1。它重写了从Vehicle继承来的description属性,提供了自己定制的包含gear信息的实现:

class Car:Vehicle{

    var gear = 1

    override var description:String{

        return super.description + " in gear \(gear)"

    }

}

let car = Car()

car.currentSpeed = 25.0

car.gear = 3

println("Car:\(car.description)")//Car:traveling at 25.0 miles per hour in gear 3

重写的description属性以super.description开头,返回了Vehicle类的description属性,并在末尾处加上了Car类的gear属性说明。



5、重写属性监视器

你可以在属性重写中为一个继承来的属性添加属性观察器。这样一来,当继承来的属性值发生改变时,你就会被通知到,无论那个属性原本是如何实现的。

注意:你不可以为继承来的常量存储型属性或继承来的只读计算型属性添加属性观察器。这些属性的值是不可以被设置的,所以,为它们提供willSet或didSet实现是不恰当。此外还要注意,你不可以同时提供重写的 setter 和重写的属性观察器。如果你想观察属性值的变化,并且你已经为那个属性提供了定制的 setter,那么你在 setter 中就可以观察到任何值变化了。


下面的例子定义了一个新类叫AutomaticCar,它是Car的子类。AutomaticCar表示自动挡汽车,它可以根据当前的速度自动选择合适的挡位。

class AutomaticCar:Car{

    override var currentSpeed:Double{

        didSet{

            gear = Int(currentSpeed / 10.0) + 1

        }

    }

}

let automatic = AutomaticCar()

automatic.currentSpeed = 35.0

println("AutomaticCar:\(automatic.description)")//AutomaticCar:traveling at 35.0 miles per hour in gear 4

当修改currentSpeed属性的时候,属性监视器didSet会给gear属性赋值。


四、防止重写

你可以通过把方法,属性或附属脚本标记为final来防止它们被重写,只需要在声明关键字前加上final特性即可。(例如:final var, final func, final class func, 以及 final subscript)

 

如果你重写了final方法,属性或附属脚本,在编译时会报错。在扩展中,你添加到类里的方法,属性或附属脚本也可以在扩展的定义里标记为 final。

 

你可以通过在关键字class前添加final关键字来将整个类标记为 final 的,这样的类是不可被继承的,否则会报编译错误。


五、总结

本章介绍了Swift中继承和重写, Swift 中的类并不是从一个通用的基类继承而来。类默认是可以被继承的。继承很简单,就是子类可以继承它父类的属性、方法和附属脚本特性。需要注意的是重写,不像OC,Swift的重写,需要明确地在类型前面加上override关键字。同样可以用super.来访问父类的属性、方法和附属脚本。需要注意几点:

1、重写属性时需要明确标明属性的类型。并且如果重写了set方法就必须提供get方法。

2、 你不可以为继承来的常量存储型属性或继承来的只读计算型属性添加属性观察器。

3、 你不可以同时提供重写的 setter 和重写的属性观察器。也就是说set和willSet以及didSet不共存。

4、类、属性、方法、附属脚本都可以通过加上final关键字来阻止被自己被重写。


参考:

1、The Swift Programming Language

2、http://www.cocoachina.com/ios/20140612/8786.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值