Kotlin - 类与构造器
发布日期:2021-06-30 18:37:44 浏览次数:2 分类:技术文章

本文共 4738 字,大约阅读时间需要 15 分钟。

类是什么?

类是一个抽象的概念,是具有某些特征的事物的概括,不特定指代任何一个具体的事物。写法:

class 
<类名>
{
<成员>
}

Number(Int、Float、Byte)、字符串(String)也是类

class Girl constructor(var nature: String,var appearance: String,var sound: String){
// 构造方法的方法体 init {
println("女孩的性格:$nature,长相:$appearance,声音:$sound") }}
  • constructor 是构造器关键字,如果只有一个构造器,则该关键字可以省略。
  • init 是构造方法的方法体,当用该类创建出一个对象时就会执行。
fun main(args: Array
) {
val girl: Girl = Girl("温柔", "甜美", "动人") // 女孩的性格:温柔,长相:甜美,声音:动人}

类构造器

构造器分为两种,分别是:

  • 主构造器:紧接在类名后面的构造器,参数可以使用 var 声明,init 是主构造器的方法体。
  • 次构造器:在类代码块中声明的构造器,参数不可以使用 var 声明,与 init 无直接关系。
class Girl constructor(var nature: String, var appearance: String, var sound: String) {
// 主构造器的方法体 init {
println("女孩的性格:$nature,长相:$appearance,声音:$sound") } // 次级构造器,无法使用 var 声明变量 constructor(nature: String, appearance: String) : this(nature, appearance, sound = "未知") {
println("次级构造器") }}fun main(args: Array
) {
val girl: Girl = Girl("温柔", "甜美") // 输出2句: // 女孩的性格:温柔,长相:甜美,声音:未知 // 次级构造器被调用}

上面说到,次级构造器与 init 无直接关系,但次级构造器会通过 :this(...) 调用主构造器,从而触发 init 方法体执行。另外,通过输出结果,可以确定 次级构造器方法体 会比 init 方法体晚执行。

个人建议:先写 init,再写次级构造器,看起来会比较舒服。

构造器省略 constructor 关键字

当没有次级构造器,只有一个主构造器时,可以省略 constructor 关键字:

class Girl(var nature: String, var appearance: String, var sound: String) {
...}

主构造器中的 var 参数

上面说到,主构造器的参数才能使用 var 声明,而次构造器则不行,那么主构造器中有无使用 var 声明的参数,会有何不同?

// 注意:只有nature使用了var声明class Girl constructor(var nature: String, appearance: String, sound: String) {
init {
println("女孩的性格:$nature,长相:$appearance,声音:$sound") } fun test() {
println(nature) // 编译OK println(appearance) // 编译不通过 println(sound) // 编译不通过 }}

init 方法体中可以访问主构造器的所有参数,而方法 test() 却只能访问 var 声明的构造器参数。

  • 主构造器中,使用 var 声明的参数,将成为成员变量,可以在类的各个方法中调用。
  • 而不使用 var 声明参数,只是临时变量,只能在 init 方法体中使用。

类的继承

子类继承父类,可以获得父类的能力,Kotlin 中使用 : 连接子父类,形成继承关系,同时,父类需要使用 open 关键字修饰:

open class Human(nature: String, appearance: String, sound: String) {
init {
println("${
this.javaClass.simpleName} 的性格:$nature,长相:$appearance,声音:$sound") }}class Girl(nature: String, appearance: String, sound: String) : Human(nature, appearance, sound)class Boy(nature: String, appearance: String, sound: String) : Human(nature, appearance, sound)

从 Java 开发者的角度来讲,Kotlin 中的类默认是 final 类,而使用 open 修饰过的类,会去除 final 关键字。

子类构造器

上述例子中,无论是 Girl 或是 Boy,它们的主构造器参数都没有使用 var 声明,实事上也不能使用 var 声明,我们可以做个实验:

// 报错:'nature' hides member of supertype 'Human' and needs 'override' modifierclass Boy(var nature: String, appearance: String, sound: String) : Human(nature, appearance, sound)

Boy 主构造器中使用 var 声明 nature,会报错提示 nature 在父类 Human 中是隐藏成员,你需要使用 override 对其修改,那就按提示把 override 关键字加上,但还会报另一个错:

// 报错:'nature' in 'Human' is final and cannot be overriddenclass Boy(override var nature: String, appearance: String, sound: String) : Human(nature, appearance, sound)

如果单单对 nature 追加 override 关键字是不够的,因为构造器中的参数默认是 final 修饰过的不可修改变量,需要对父类 Human 主构造器中的 nature 参数再追加 open 关键字来去除这个限制。

open class Human(open var nature: String, var appearance: String, var sound: String) {
...}class Boy(override var nature: String, appearance: String, sound: String) : Human(nature, appearance, sound)

好了,终于把子类 Boy 的主构造器参数 nature 加上了 var 声明,那这有什么呢?答案是似乎没啥用。

因为就算子类 Boy 主构造器参数 nature 不用 var 声明,也一样可以在子类 Boy 的成员方法中使用这个 nature 变量,原因是 nature 早已在父类 Human 的主构造器中使用了 var 声明,它是父类 Human 的成员变量(类似 Java 中的 protected final String nature ),可以被子类 Boy 识别和调用。所以,话说回来,我们什么时候会需要使用 var 来声明与父类构造器参数同名的子类主构造器参数呢?

open class Human(nature: String, var appearance: String, var sound: String) {
...}class Boy(var nature: String, appearance: String, sound: String) : Human(nature, appearance, sound)

当父类主构造器参数没有使用 var 声明,而子类又需要在成员方法中使用该参数时,可以在子类的构造器中,为该参数使用 var 追加声明,让其成为子类的成员变量。除此之外,子类主构造器参数的使用与普通的主构造器参数没什么区别。

Any 类

在 Kotlin 中,Any 是所有类的始祖,相当于 Java 中的 Object。

/** * The root of the Kotlin class hierarchy. Every Kotlin class has [Any] as a superclass. */public open class Any {
...}

包(Package)

  • 包就是命名空间
  • 包的声明必须在非注释代码的第一行
  • 类的全名是 包名+类名 ,如:com.area.guangzhou.Human

举个例子,在 com.area.guangzhou 包下声明一个 Human 类:

package com.area.guangzhou // 第一行代码:包声明/** * 广州人 */class Human {
}

由于包的存在,因此工程中可以在不同的目录下,声明多个类名相同的类,比如在 com.area.shantou 再声明一个 Human 类:

package com.area.shantou/** * 汕头人 */class Human {
}

因为类名相同,所以在同个文件中使用时,为了区分 2 个 Human,需要使用类的全名:

fun main(args: Array
) {
val gzHuman = com.area.guangzhou.Human() val stHuman = com.area.shantou.Human()}

这样写出来的代码,看起来很不雅观,这时我们可以使用 as 关键字对导包设置别名:

import com.area.guangzhou.Human as 自给人import com.area.shantou.Human as 胶己人fun main(args: Array
) {
val gzHuman = 自给人() val stHuman = 胶己人()}

注意:这里的代码仅为演示说明,实际开发中,为了避免不必要的麻烦,最好不要使用中文进行 coding。

转载地址:https://linqiarui.blog.csdn.net/article/details/112972291 如侵犯您的版权,请留言回复原文章的地址,我们会给您删除此文章,给您带来不便请您谅解!

上一篇:Kotlin - 空类型和智能类型转换
下一篇:Kotlin - 数据类型

发表评论

最新留言

很好
[***.229.124.182]2024年04月25日 08时11分54秒

关于作者

    喝酒易醉,品茶养心,人生如梦,品茶悟道,何以解忧?唯有杜康!
-- 愿君每日到此一游!

推荐文章

数据挖掘与数据分析(三)—— 探索性数据分析EDA(多因子与复合分析) & 可视化(1)—— 假设检验(μ&卡方检验&方差检验(F检验))&相关系数(皮尔逊&斯皮尔曼) 2019-04-30
RRT算法(快速拓展随机树)的Python实现 2019-04-30
路径规划(二) —— 轨迹优化(样条法) & 局部规划(人工势能场法) & 智能路径规划(生物启发(蚁群&RVO) & 强化学习) 2019-04-30
D*算法 2019-04-30
强化学习(四) —— Actor-Critic演员评论家 & code 2019-04-30
RESTful API 2019-04-30
优化算法(四)——粒子群优化算法(PSO) 2019-04-30
数据挖掘与数据分析(三)—— 探索性数据分析EDA(多因子与复合分析) & 可视化(2)——回归分析(最小二乘法&决定系数&残差不相关)&主成分分析&奇异值分解 2019-04-30
数据在Oracle中的存储 2019-04-30
优化算法(五)—人工蜂群算法Artificial Bee Colony Algorithm(ABC) 2019-04-30
轨迹规划 trajectory planning 2019-04-30
AGV自动导引运输车 2019-04-30
Trie树(字典树) 2019-04-30
COMP7404 Machine Learing——Logistic Regression 2019-04-30
COMP7404 Machine Learing——Regularization(参数C) 2019-04-30
COMP7404 Machine Learing——KNN 2019-04-30
COMP7404 Machine Learing——SVM 2019-04-30
COMP7404 Machine Learing——Decision Tree & Random Forests 2019-04-30
COMP7404 Machine Learing——Hyperparameter Grid Search & Nested Cross-Validation 2019-04-30
COMP7404 Machine Learing——Confusion Matrix & Precision/Recall/F1 2019-04-30