class Person @Inject constructor(name: String){/…/}
我们也可以直接在主构造方法里面声明类的属性:
class Person (val firstProperty: String, val secondProperty: Int) {}
而且,在 1.4 及以上版本 Kotlin 中,我们可以在最后一个属性变量后面加个小尾巴逗号(trailing comma):
class Person (
val firstProperty: String,
val secondProperty: Int,
) {}
函数的参数列表也可以用:
fun reformat (
str: String,
secondProperty: Int = 0,
) {}
或者:
val colors = listOf (
‘red’,
‘greedn’,
)
这个小尾巴的作用是什么呢?首先,每个参数后面都有逗号,就避免最后一个参数不带参数这种特殊情况的出现,添加新参数或者调整参数顺序时可以连着参数后面的逗号一起进行;其次,当我们在参数列表最后添加带有逗号的新参数时,在查看代码 diff 的时候,只会出现一行新增代码,而不是一行删除的代码+一行新增的代码,具有更好的可阅读性。
注意,虽然叫构造方法,但是主构造方法里面不能含有代码行。初始化代码要放在初始化块里面,以 init
作为关键字,如果存在多个初始化块,按照其出现的顺序依次执行,用法如下:
class Person constructor(name: String) {
val firstProperty = name.toUpperCas
【一线大厂Java面试题解析+核心总结学习笔记+最新架构讲解视频+实战项目源码讲义】
浏览器打开:qq.cn.hn/FTf 免费领取
e()
init {
println(“firstProperty is ${name}”)
}
val secondProperty = “secondProperty: ${name.length}”.also(::println)
init {
println(“secondProperty is ${name.length}”)
}
}
可以看到,init
块可以和属性声明交叉进行。
[](
)副构造方法
出现在类名称后面的 constructor() 叫主构造方法,而出现在类内部的 constructor() 我们称之为副构造方法:
class Person {
var children: MutableList = mutableListOf()
constructor(parent : Person) {
parent.children.add(thisg)
}
}
主副构造方法以及init
块之间有什么关系吗?首先,init
块里面的代码都会成为主构造方法的一部分。然后,副构造方法都必须直接或间接的委托给主构造方法,即主构造方法体在副构造方法之前执行。即使类没有声明主构造方法,依然如此,只不过这种委托是隐式发生的。
class Constructors {
init {
println(“init block”)
}
constructor(i : Int) {
println(constructor $i")
}
}
[](
)实例化
跟 Java 不同,Kotlin 中实例化对象无需 new 关键字:
val person = Person(“Linus”)
[](
)属性
详见 [Kotlin 属性](
)。
[](
)继承
Kotlin 类默认是不能继承的,只有被 open 修饰的类才能被继承:
open class Base
这是一个比 Java 方便的地方。根据笔者的经验,多数时候我们是不想类被别人继承的。特别是在多业务线开发的时候,你写了一个控价,本来这个控件是业务定制的,但是总有一些想偷懒的同学直接继承你的控件来实现他们的业务需求,连声招呼都不带打的,当你需要迭代自己的控件时,发现因为继承关系的存在导致各种问题。而往往写 Java 代码时会忽略使用 final 关键字,Kotlin 就解决了这个痛点,默认不能继承,只有显式使用 open 关键字才能被继承。
如果 Kotlin 子类含有主构造方法,那么基类也要显式的展示出其主构造方法,而且二者的参数要一致:
open class Base
class Derived(p : Int) : Base§
如果子类只有副构造方法,没有主构造方法,那么子类中的副构造方法要使用冒号来继承父类中对应的构造方法,或者委托给其他的副构造方法:
class MyView : View {
constructor (ctx : Context) : super(ctx)
constructor (ctx : Context, attrs: AttributeSet) : super(ctx, attrs)
}
[](
)方法重写
与 Java 不同,在 Koltin 类中被 open 修饰的方法才能被重写。没有被 open 修饰的方法,不能在子类里面声明同样名称的方法,不然编译器会报错,比如下面的 Shape.fill() 方法:
open class Shape {
open fun draw() {/…/}
fun fill() {/…/}
}
class Circle() : Shape() {
override fun draw() {/…/}
}
注意,被 override 修饰的方法是天然 open 的,如果不想被重写,可以使用 final 关键字:
open class Rectangle() : Shape() {
final override fun draw() {/…/}
}
那如果我在非 open 的类的方法前面添加 open 修饰符会怎么样?比如:
class NonOpenClass {
open fun demo() {/…/}
}
此时,open 修饰符不会生效。
[](
)属性重写
跟方法重写类似,子类重写父类的属性,也是使用 override 关键字来实现。同时二者的类型要兼容:
open class Shape {
open val vertextCount: Int = 0
}
open class Rectangle() : Shape() {
override val vertextCount = 4
}
在子类中可以用 var 类型的属性重写父类中 val 类型的属性,而反过来则不行。这是因为,val 类型的属性本质上声明了一个 get 方法,而使用 var 重写它相当于新增了一个 set 方法。
注意,我们也可以使用 override 关键字来修饰主构造方法中的属性:
interface Shape {
val vertextCount: Int