每天学一点 Kotlin -- 类的构造函数

----《第一季Kotlin崛起:次世代Android开发 》学习笔记

总目录:每天学一点 Kotlin ---- 目录
上一篇:每天学一点 Kotlin -- 初识类和对象
下一篇:每天学一点 Kotlin -- 类的属性

1. 构造函数

1.1 构造函数是一种特殊的方法。主要用来在创建对象时初始化对象, 即为对象成员变量赋初始值。

1.2 在 Kotlin 中一个类可以有一个主构造函数以及一个或多个次构造函数。

2. 主构造函数

2.1 主构造函数是类头的一部分:它跟在类名(与可选的类型参数)后。
第一种写法:

class Friend constructor(val name: String, val age: Int){}

第二种写法:

class Friend constructor(name: String, age: Int){
   val name: String = name
   val age: Int = age
}

2.2 如果主构造函数没有任何注解或者可见性修饰符,可以省略 constructor 关键字。写成:

class Friend(name: String, age: Int){
   val name: String = named
   val age: Int = age
}

注意:如果主构造函数中有注解或者可见性修饰符,那么 constructor 关键字是不能省略的。比如:

class Friend private constructor(val name: String, val age: Int){}

2.3 和 Java 中的区别: 在 Java 中,Friend(name: String, age: Int)这个构造函数是写在 Friend 类中的,和成员方法是平行的。在 Kotlin 中,这个构造函数是在定义类的开头就写出来的。

2.4 类定义紧跟的大括号里的内容是对这个类的详细声明部分,可以定义属性和方法,定义的属性和方法是不属于构造函数的,主构造函数本身是没有代码的。 ---- 这点很绕人,想不通,个人觉得不是很好,分明就是类的声明么,为啥要称为主构造函数呢,傻逼。。。

3. Init 代码块

3.1 在主构造函数中不能有代码,Kotlin 中提供了初始化代码块,那就是 init。

3.2 在 init 代码块中,可以编写这个对象在实例化构造完成后要执行的语句。

3.3 举个栗子:

fun main() {
    var friend = Friend2("xiaoMing", 20)
}

class Friend2(name: String, age: Int) {
    init {
        println("name: $name , age = $age")
    }

    var name: String = ""
    var age: Int = 0
}

打印结果:

name: xiaoMing , age = 20

可以看到,实例化对象之后,init 会被自动调用,这才跟 Java 中构造函数有一点像了。

4. 次构造函数

4.1 在类定义中,用 constructor 关键字来声明次构造函数

4.2 按照 Java 中的习惯,我们定义以下类:

class Friend3(name: String, age: Int) {
    var name: String = name
    var age: Int = age

    init {
        println("name: $name , age = $age")
    }

    constructor(name: String, age: Int) {
        this.name = name
        this.age = age
    }
}

但是上面的写法是错误的,编译器会在主构造函数和次构造函数的下面画出红线,那我们把主构造函数的参数删除了, 并且把类中的成员变量重新赋值:

fun main() {
    val friend4 = Friend4("xiaoMing", 30)
}

class Friend4 {
    var name: String = "null"
    var age: Int = 0

    init {
        println("name: $name , age = $age")
    }

    // 次构造函数
    constructor(name: String, age: Int) {
        this.name = name
        this.age = age
    }
}

打印结果:

name: null , age = 0

4.3 小结:
(1) init 初始化块跟 Java 中静态初始化块很相似。
(2) constructor 声明的次构造函数才跟 Java 中构造函数是相似了。。。

4.4 主构造函数和次构造函数可以共存:前提是每个次构造函数都需要委托给主构造函数。举个栗子1:

fun main() {
    var friend5_1 = Friend5("xiaoMing", 51)
    var friend5_2 = Friend5("xiaoQiang", 52, friend5_1)
}

class Friend5(name: String, age: Int) {
    var friend: Friend5? = null

    init {
        println("name = ${name}, age = ${age}")
    }

    constructor(name: String, age: Int, friend: Friend5) : this(name, age) {
        friend.friend = this
    }
}

打印结果:

name = xiaoMing, age = 51
name = xiaoQiang, age = 52

举个栗子2:

fun main() {

    val baby1 = Person("baby1", 3)
    val baby2 = Person("baby2", 4)
    val parent1 = Person("parent1", 40)
    val parent2 = Person("parent2", 50)
    val child1 = Person("child1", 20, mutableListOf(parent1, parent2), mutableListOf(baby1, baby2))

    baby1.showPersonInfo()

    parent1.showPersonInfo()
    parent1.showChildrenInfo()

    child1.showPersonInfo()
    child1.showChildrenInfo()
    child1.showParentInfo()
}

class Person(name: String, age: Int) {

    val name: String = name
    val age: Int = age
    var parents = mutableListOf<Person>()
    var children = mutableListOf<Person>()

    // 次构造函数
    constructor(name: String, age: Int, parents: MutableList<Person>, children: MutableList<Person>) : this(name, age) {
        parents.forEach { it.children.add(this) }
        children.forEach { it.parents.add(this) }
        this.parents.addAll(parents)
        this.children.addAll(children)
    }

    fun showPersonInfo() = println("name = ${this.name}, age = ${this.age}")

    fun showChildrenInfo() = children.forEach { it.showPersonInfo() }

    fun showParentInfo() = parents.forEach { it.showPersonInfo() }
}

打印结果:

name = baby1, age = 3
name = parent1, age = 40
name = child1, age = 20
name = child1, age = 20
name = baby1, age = 3
name = baby2, age = 4
name = parent1, age = 40
name = parent2, age = 50

5. 默认构造函数

5.1 当我们没有任何主构造函数和次构造函数的时候,Kotlin 会给我们自动生成一个默认的主构造函数,也就是没有任何构造参数。

5.2 在之前的内容里面,下面这一句就使用的是默认构造函数:

val friend = Friend()
相关代码:https://gitee.com/fzq.com/test-demo
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,390评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,821评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,632评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,170评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,033评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,098评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,511评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,204评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,479评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,572评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,341评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,893评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,171评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,486评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,676评论 2 335

推荐阅读更多精彩内容