1.延迟初始化属性
// Kotlin可以对属性设置为延迟初始化
lateinit var dept: Department
提示: 延迟初始化属性要求:不能是可空类型;只能使用为var声明;lateinit关键字应该放在var之前。
2. 委托属性
Kotlin提供一种委托属性,使用by关键字声明
class User {
var name: String by Delegate()
}
class Delegate {
operator fun getValue(thisRef: Any, property: KProperty<*>): String = property.name
operator fun setValue(thisRef: Any?, property: KProperty<*>, value: String) {
println(value)
}
}
fun main(args: Array<String>) {
val user = User()
user.name = "Tom"
println(user.name)
}
注意这两个函数前面都有operator关键字修饰,operator所修饰的函数是运算符重载函数,本例中说明了getValue和setValue函数重载by运算符。
3.惰性加载属性
惰性加载属性与延迟初始化属性类似,只有第一次访问该属性时才进行初始化。不同的是惰性加载属性使用的lazy函数声明委托属性,而延迟初始化属性lateinit关键字修饰属性。还有惰性加载属性必须是val的,而延迟初始化属性必须是var的。
val fullName: String by lazy { // ①
firstName + "." + lastName
}
lateinit var dept: Department // ②
上述代码第①行声明了的惰性加载属性fullName,by后面是lazy函数,注意lazy不是关键字,而是函数。lazy函数后面跟着的是尾随Lambda表达式。惰性加载属性使用val声明。
代码第②行声明了延迟初始化属性dept,使用关键字lateinit。延迟初始化属性使用var声明。
4. 可观察属性
另一个使用委托属性示例是可观察属性,委托对象监听属性的变化,当属性变化时委托对象会被触发。
class Department {
var no: Int = 0 // 部门编号属性
var name: String by Delegates.observable("<无>") { p, oldValue, newValue -> ①
println("$oldValue -> $newValue")
}
}
fun main(args: Array<String>) {
val dept = Department()
dept.no = 20
dept.name = "技术部" //输出<无> -> 技术部 ②
dept.name = "市场部" //输出技术部 -> 市场部 ③
}
上述代码第①行是声明name委托属性,by关键字后面Delegates.observable()函数有两个参数:第一个参数是委托属性的初始化值,第二个参数是属性变化事件的响应器,响应器是函数类型,具体调用时可使用Lambda表达式作为实际参数。在用Lambda表达式中有三个参数,其中p是属性,oldValue是属性的旧值,newValue是属性的新值。
5. 扩展函数
fun 接收类型.函数名(参数列表) : 返回值类型 {
函数体
return 返回值
}
//基本数据类型扩展
fun Double.interestBy(interestRate: Double): Double { ①
return this * interestRate
}
//自定义账户类
class Account {
var amount: Double = 0.0 //账户金额
var owner: String = "" //账户名
}
//账户类扩展函数
fun Account.interestBy(interestRate: Double): Double { ②
return this.amount * interestRate
}
6. 扩展属性
var|val 接收类型.属性名 [ : 数据类型]
[getter访问器]
[setter访问器]
7. 定义中缀运算符
注意 定义中缀运算符,就是要声明一个infix关键字修饰的函数,该函数只能有一个参数,该函数不能是顶层函数,只能成员函数或扩展函数。
//定义中缀函数interestBy
infix fun Double.interestBy(interestRate: Double): Double {
return this * interestRate
}
8. 主构造函数
主构造函数涉及到两个关键字constructor和init。主构造函数在类头中、类名的后面声明,使用关键字constructor。
class Rectangle constructor(w: Int, h: Int) {
// 矩形宽度
var width: Int
// 矩形高度
var height: Int
// 矩形面积
var area: Int
init { //初始化代码块
width = w
height = h
area = w * h// 计算矩形面积
}
}
// 简化
class Rectangle constructor(var width: Int, var height: Int) {
// 矩形面积
var area: Int
init {//初始化代码块
area = width * height// 计算矩形面积
}
}
如果所有的属性都在主构造函数中初始化,可以省略init代码块
提示: 如果主构造函数没有注解(Annotation)或可见性修饰符,constructor关键字可以省略。
class User(val name: String, var password: String)
// User类不能省略constructor关键字,因为User前面private可见行修饰符
class User private constructor(val name: String, var password: String)
9. 次构造函数
由于主构造函数只能有一个,而且初始化时只有init代码块,有时候不够灵活,这时可以使用次构造函数。次构造函数是在函数体中声明的,使用关键字constructor声明。
class Rectangle(var width: Int, var height: Int) {
// 矩形面积
var area: Int
init {//初始化代码块
area = width * height// 计算矩形面积
}
constructor(width: Int, height: Int, area: Int) : this(width, height) {
this.area = area
}
constructor(area: Int) : this(200, 100) {//width=200 height=100
this.area = area
}
}
次构造函数后面的this(width, height)和this(200, 100)表达式就是调用当前对象的主构造函数
10. 默认构造函数
如果一个非抽象类中根本看不到任何的构造函数,编译器会为其生成一个默认的构造函数,即无参数public的主构造函数。
//默认构造函数
class User {
// 用户名
val username: String?
// 用户密码
val password: String?
init {
username = null
password = null
}
}
11.可见性修饰符使用规则
可见性 修饰符 类成员声明 顶层声明 说明
公有 public 所有地方可见 所有地方可见 public是默认修饰符
内部 internal 模块中可见 模块中可见 不同于Java中的包
保护 protected 子类中可见 顶层声明中不能使用
私有 private 类中可见 文件中可见
12. 数据类
// 数据类的声明很简单,只需要类头class前面加上data关键字即可
data class User(val name: String, var password: String)
提示: 使用data声明的数据类的主构造函数中参数一定声明为val或var的,不能省略。而普通类可以省略的,
使用is和!is进行类型检查
使用as和as?进行类型转换
Lambda体中it隐式变量是由Kotlin编译器生成的,它的使用有两个前提:一是Lambda表达式只有一个参数,二是根据上下文能够推导出参数类型。
泛型
泛型特性对Kotlin影响最大是在集合中使用泛型
Kotlin中创建对象数组有三种方式:
arrayOf(vararg elements: T)工厂函数。指定数组元素列表创建元素类型为T的数组,vararg表明参数个数是可变的。
arrayOfNulls<T>(size: Int)函数。size参数指定数组大小,创建元素类型为T的数组,数组中的元素为空值。
Array(size: Int, init: (Int) -> T)构造函数。通过size参数指定数组大小,init参数指定一个用于初始化元素的函数,实际使用时经常是Lambda表达式。
基本数据类型数组的创建三种方式,下面以Int类型为例介绍一下:
intArrayOf(vararg elements: Int)工厂函数。通过对应的工厂函数,vararg表明参数是可变参数,是Int数据列表。
IntArray(size: Int)构造函数。size参数指定数组大小创建元素类型为Int的数组,数组中的元素为该类型默认值,Int的默认值是0。
IntArray(size: Int, init: (Int) -> Int)构造函数。通过size参数指定数组大小,init参数指定一个用于初始化元素的函数,参数经常使用Lambda表达式。
Kotlin 数据类
data class User(var name: String, var password: String)
Kotlin数据类,其中有两个属性,var声明的属性会生成setter和getter函数,如果是val声明的属性是只读的,只生成getter函数
引用类有两种形式:类名::class和对象::class。
//1.获得“类名::class”引用类
val clz1 = Int::class
val clz2 = Person::class
val person = Person("Tom")
//2.获得“对象::class”引用类
val clz3 = person::class