宗旨补充,类和继续

时间:2019-10-08 00:20来源:编程技术
官方文档: 子父类中的函数 当子类出现父类一模一样的函数时, 子类对象调用该函数,会运行子类函数的内容。 如同父类的函数被覆盖一样。 这种情况是函数的另一个特性:重写(覆

官方文档:

子父类中的函数
当子类出现父类一模一样的函数时,
子类对象调用该函数,会运行子类函数的内容。
如同父类的函数被覆盖一样。
这种情况是函数的另一个特性:重写(覆盖)

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

当子类继承父类,沿袭了父类的功能到子类中,
但是子类虽具备该功能,但功能的内容和父类不一致,
这时,没有表定义新功能,而是使用覆盖特性,保留父类的功能定义,并重写功能内容。

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) { } 与普通变量一样,主构造函数中声明的类属性可以是可变或只读

覆盖:
1、子类覆盖父类,必须保证子类权限大于等于父类权限才可以覆盖,否则编译失败。
2、静态只能覆盖静态

2.次构造函数

在类体中也可以声明前缀有 constructor 的次构造函数: class Person { constructor(parent: Person) { parent.children.add } } 1.如果类有一个主构造函数,每个次构造函数需要委托给主构造函数,可以直接委托或者通过别的次构造函数间接委托委托到同一个类的另一个构造函数用 this 关键字即可: class Person(val name: String) { constructor(name: String, parent: Person) : this { parent.children.add } }2.如果非抽象类没有声明任何构造函数, 会自动生成无参数的主构造函数, 可见性是public如果不想构造函数是public,需添加 private constructor: class DontCreateMe private constructor () { }

重载:只看同名函数的参数列表。
重写:子父类方法要一模一样。

3.创建类的实例对象

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

子父类出现后,类成员的特点:
类中成员:
1.变量
2.函数
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 class Derived : Base2.初始化超类的构造函数如果类具有主构造函数,则用主构造函数的参数初始化基类构造函数如果类没有主构造函数,则每个次构造函数必须使用super关键字初始化基类型,或委托给另一构造函数 class MyView : View { constructor(ctx: Context) : super constructor(ctx: Context, attrs: AttributeSet) : super(ctx, attrs) }

变量
如果子类中出现非私有的同名成员变量时,
子类要访问本类中的变量,用this
子类要访问父类中的同名变量,用super

2.覆盖父类方法

与Java不同,Kotlin需显式标注可覆盖的成员和覆盖后的成员 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() {} }

super和this的使用几乎一样
this代表的是本类对象的引用
super代表的是父类对象的引用

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

子父类的构造函数
在对子类对象进行初始化时,父类的构造函数也会运行,
因为子类的构造函数默认第一行有一条隐式的语句 super();
super():会访问父类中空参数的构造函数,而且子类中所有的构造函数默认第一行都是super();

4.覆盖规则

在 Kotlin 中,实现继承由下述规则规定:如果类从直接超类继承相同成员的多个实现,它必须覆盖这个成员并提供其自己的实现: open class A { open fun f() { print } } interface B { fun f() { print } // 接口默认是open } class C, B { override fun f() { super<A>.f() // 调用 A.f() super<B>.f() // 调用 B.f() } }

简书: :

为什么子类一定要访问父类中的构造函数
因为父类中的数据,子类可以直接获取,所以子类对象在建立时,需要先查看父类是如何对这些数据进行初始化的。
所以子类只按对象初始化时,要先访问以下父类中的构造函数。
如果要访问父类中指定的构造函数,可以通过手动定义super语句的方式来指定。

注意:super语句一定要定义在子类构造函数的第一行。

子类实例化过程
结论:
子类的所有的构造函数,默认都会访问父类中空参数的构造函数。
因为子类每一个构造函数内的第一行都有一句隐式super();
当父类总没有空参数的构造函数时,子类必须手动super或者this语句形式来指定要访问的构造函数。
当然,子类的构造函数第一行也可以手动指定this语句来访问本类中的构造函数。
子类中至少会有一个构造函数会访问父类中的构造函数。

编辑:编程技术 本文来源:宗旨补充,类和继续

关键词:

  • 上一篇:没有了
  • 下一篇:没有了