Kotlin-08.类和继承(class/Inheritance)

官方文档: http://kotlinlang.org/docs/reference/classes.html

1.定义类

和java一样,Kotlin 中使用关键字 class 声明/定义类
    class MyClass(param: type) {
    }
类声明由类名、类头(主构造函数参数)和类体构成, 类头和类体都是可选的    
如果没有类体,可以省略花括号:
    class MyClass

2.构造函数

1.主构造函数

在 Kotlin 中的一个类可以有一个主构造函数、多个次构造函数!
主构造函数是类头的一部分,由类名(可选参数)构成
    class Person constructor(name: String) {
    }

1.省略 constructor 关键字
    如果主构造函数没有注解或者可见性修饰符(public/private等),可以省略 constructor 关键字
    class Person(name: String) {
    }
    如果构造函数有注解或可见性修饰符,必需要 constructor 关键字:
    class Person public @Inject constructor(name: String) {
    }

2.主构造函数的初始化代码可放到 init 初始化块中:
    class Person(name: String) {
        init {
            // 主构造函数参数可以在初始化块中使用
            logger.info("value: ${name}")
        }
        // 主构造函数参数也可以在属性初始化器中使用
        val key = name.toUpperCase()
    }

3.在主构造函数中声明类属性(类成员变量/字段)以及初始化属性:
    class Person(val firstName: String, val lastName: String, var age: Int) {
    }
    与普通变量一样,主构造函数中声明的类属性可以是可变(var)或只读(val)

2.次构造函数

在类体中也可以声明前缀有 constructor 的次构造函数:
    class Person {
        constructor(parent: Person) {
            parent.children.add(this)
        }
    }
    
1.如果类有一个主构造函数,每个次构造函数需要委托给主构造函数,可以直接委托或者通过别的次构造函数间接委托
委托到同一个类的另一个构造函数用 this 关键字即可:
    class Person(val name: String) {
        constructor(name: String, parent: Person) : this(name) {
            parent.children.add(this)
        }
    }

2.如果非抽象类没有声明任何构造函数, 会自动生成无参数的主构造函数, 可见性是public
如果不想构造函数是public,需添加 private constructor:
    class DontCreateMe private constructor () {
    }

3.创建类的实例对象

和java不同,Kotlin没有new关键字, 创建一个类的实例对象:
    val person = Person("lioil")    

3.继承

1.定义

和java类似,在 Kotlin 中所有类都有一个共同的超类(基类/父类) Any
    class Example // 从 Any 隐式继承
    Any 不是 java.lang.Objec, Any除了equals()、hashCode()和toString()外没有其它任何成员!

1.声明一个显式超类(基类/父类):
open表示允许其它类继承, 和Java中final相反, 默认情况下,Kotlin所有类都是final
    open class Base(p: Int)
    class Derived(p: Int) : Base(p)

2.初始化超类(基类/父类)的构造函数
如果类具有主构造函数,则用主构造函数的参数(并且必须)初始化基类构造函数
如果类没有主构造函数,则每个次构造函数必须使用super关键字初始化基类型,或委托给另一构造函数
    class MyView : View {
        constructor(ctx: Context) : super(ctx)
        constructor(ctx: Context, attrs: AttributeSet) : super(ctx, attrs)
    }

2.覆盖父类方法

与Java不同,Kotlin需显式标注可覆盖的成员(open)和覆盖后的成员(override)
    open class Base {
        open fun v() {} // 如果没标注open, 则子类不允许定义相同名函数
        fun nv() {}
    }

    class Derived() : Base() {
        // 如果没标注override, 则编译器将会报错            
        override fun v() {}  
    }
    
标记 override 相当于open,可在子类中覆盖, 如果要禁止再次覆盖,要final 关键字:
    open class AnotherDerived() : Base() {
        final override fun v() {}
    }

3.覆盖父类属性

覆盖属性与覆盖方法类似,
在父类的属性必须以 override 开头,并且父子类必须具有兼容的类型
每个声明的属性可由具有初始化器的属性或者具有getter方法的属性覆盖
    open class Foo {
        open val x: Int get { …… }
    }

    class Bar1 : Foo() {
        override val x: Int = ……
    }

可用var属性覆盖val属性,但反之则不行,
因为val属性本质是只声明一个getter方法,而将其覆盖为var,只是在子类中添加一个setter方法

可在主构造函数中使用 override 关键字覆盖父类的属性:
    interface Foo {
        val count: Int
    }

    class Bar1(override var count: Int) : Foo

4.覆盖规则

在 Kotlin 中,实现继承由下述规则规定:
如果类从直接超类继承相同成员的多个实现,它必须覆盖这个成员并提供其自己的实现:
    open class A {
        open fun f() { print("A") }
    }

    interface B {
        fun f() { print("B") } // 接口默认是open
    }

    class C() : A(), B {       
        override fun f() {
            super<A>.f() // 调用 A.f()
            super<B>.f() // 调用 B.f()
        }
    }

简书: http://www.jianshu.com/p/2f9554932f74 CSDN博客: http://blog.csdn.net/qq_32115439/article/details/73442373
GitHub博客:http://lioil.win/2017/06/18/Kotlin-classes.html
Coding博客:http://c.lioil.win/2017/06/18/Kotlin-classes.html