解析Swift语言面相对象编程中的继承特性

取大于形态的能力被定义为继承。一般一个类可以从另一个类继承属性和方法。类可以进一步划分到子类和超类。

子类:当一个类从另一个类继承属性,方法和功能被称为子类

超类:类包含属性,方法和功能被其它类继承称为超类

Swift 中类包含父类和调用访问方法,属性,功能和重写方法。另外,属性观察者也用于添加属性和修改所存储的或计算的特性的方法。

基类
一个类如果不从其它类继承方法,属性或功能,那么它被称为“基类”。

classStudDetails{var stname:String!var mark1:Int!var mark2:Int!var mark3:Int!     init(stname:String, mark1:Int, mark2:Int, mark3:Int){self.stname = stname         self.mark1 = mark1         self.mark2 = mark2         self.mark3 = mark3     }}let stname ="swift"let mark1 =98let mark2 =89let mark3 =76

println(stname) println(mark1) println(mark2) println(mark3)


当我们使用 playground 运行上面的程序,得到以下结果。

swift
98
89
76

这里 StudDetails 类被定义为基类,它用于包含学生名字和三个科目标记为:mark1, mark2 和 mark3. 'let'关键字在 playground 中初始化并使用 “println” 函数打印显示基础类的值。

子类
在现有的基类上定义一个新的类就叫作“子类”。子类继承了其基类的属性,方法和功能。要定义一个子类使用 “ : ” 在基类名称前。

classStudDetails{var mark1:Int;var mark2:Int;         init(stm1:Int, results stm2:Int){         mark1 = stm1;         mark2 = stm2;}         func print(){         println("Mark1:\(mark1), Mark2:\(mark2)")}}class display :StudDetails{     init(){super.init(stm1:93, results:89)}}let marksobtained = display() marksobtained.print()


当我们使用 playground 运行上面的程序,得到以下结果。

Mark1:93, Mark2:89

“StudDetails” 类定义为学生标记声明的超类以及子类的 'display' 从它的超类继承以打的标记。子类定义学生标记和调用打印方法来显示学生的标志。

覆盖/重写
访问超类的实例,类型方法,例如,类型属性和下标子类提供覆盖的概念。 'override' 关键字用来覆盖超类中声明的方法。

访问超级类的方法,属性和下标
“super”关键字作为前缀用来访问超类中声明的方法,属性和下标。


方法和属性覆盖 方法覆盖
继承实例和类型的方法可以通过 'override' 关键字覆盖在子类中定义的方法。在这里,在子类中重写打印来访问超类打印type属性。

class cricket {     func print(){         println("Welcome to Swift Super Class")}}class tennis: cricket  {override func print(){         println("Welcome to Swift Sub Class")}}let cricinstance = cricket() cricinstance.print()let tennisinstance = tennis() tennisinstance.print()


当我们使用 playground 运行上面的程序,得到以下结果。

Welcome to Swift Super Class
Welcome to Swift Sub Class

属性重写
可以覆盖继承的实例或类属性来提供自定义的getter和setter 属性,或添加属性观察者,当下层属性值更改时以使重写属性到观察者。

重写属性getter和setter
Swift 允许用户提供自定义 getter和setter 覆盖继承的属性,无论是存储还是计算属性。子类不知道继承的属性名称和类型。因此,至关重要的是,用户需要在子类中指定,名称和在超类中指定重写属性的类型。

这可以通过两种方式来完成:

当 setter 被定义为重写属性,用户必须也要定义 getter。

当我们不希望修改继承属性的getter,我们可以通过简单的语法“super.someProperty”来给超类继承值。

classCircle{var radius =12.5var area:String{return"of rectangle for \(radius) "}}classRectangle:Circle{varprint=7overridevar area:String{returnsuper.area +" is now overridden as \(print)"}}let rect =Rectangle() rect.radius =25.0 rect.print=3 println("Radius \(rect.area)")


当我们使用 playground 运行上面的程序,得到以下结果。

Radius of rectangle for 25.0 is now overridden as 3

重写属性观察者
当一个新的属性需要为继承的属性被添加,在 Swift 中推出 “属性重写” 的概念。通知用户当继承属性值被更改。但重写不适用于继承的常量存储属性和继承只读计算属性。

classCircle{var radius =12.5var area:String{return"of rectangle for \(radius) "}}classRectangle:Circle{varprint=7overridevar area:String{returnsuper.area +" is now overridden as \(print)"}}let rect =Rectangle() rect.radius =25.0 rect.print=3 println("Radius \(rect.area)")classSquare:Rectangle{overridevar radius:Double{         didSet {print=Int(radius/5.0)+1}}}let sq =Square() sq.radius =100.0 println("Radius \(sq.area)")


当我们使用 playground 运行上面的程序,得到以下结果。

Radius of rectangle for 25.0 is now overridden as 3
Radius of rectangle for 100.0 is now overridden as 21

最终属性以防止重写
当用户不需要让别人访问超类的方法,属性或下标,Swift 引入“final”属性,以防止覆盖。 当 “final” 属性被声明后,将不允许超类的方法,属性和下标被覆盖。在超类不可以有 'final' 。当 “final” 属性被声明后,用户限制子类创建。

 finalclassCircle{finalvar radius =12.5var area:String{return"of rectangle for \(radius) "}}classRectangle:Circle{varprint=7overridevar area:String{returnsuper.area +" is now overridden as \(print)"}}let rect =Rectangle() rect.radius =25.0 rect.print=3 println("Radius \(rect.area)")classSquare:Rectangle{overridevar radius:Double{         didSet {print=Int(radius/5.0)+1}}}let sq =Square() sq.radius =100.0 println("Radius \(sq.area)")


当我们使用 playground 运行上面的程序,得到以下结果。

<stdin>:14:18: error: var overrides a 'final' var     override var area: String {                  ^ <stdin>:7:9: note: overridden declaration is here     var area: String {         ^ <stdin>:12:11: error: inheritance from a final class 'Circle'     class Rectangle: Circle {           ^ <stdin>:25:14: error: var overrides a 'final' var override var radius: Double {              ^ <stdin>:6:14: note: overridden declaration is here    final var radius = 12.5


当超类声明为 “final”和数据类型也被宣明为'final',程序将不允许再创建子类,否则它会引发错误。