- 开发环境:IntelliJ IEDA
- 个人博客:http://blog.csdn.net/IInmy
- 项目源码:https://github.com/Rushro2m/KotlinForOfficial
- 官方文档中文版:https://www.kotlincn.net/docs/reference/properties.html
1、声明属性
Kotlin的类可以有属性。属性可以用关键字var
声明为可变的,否则使用只读关键字val
class Address {
var name: String = "张三"
var street: String = "泉口街道"
var city: String = "北京"
var state: String = "未压缩"
var zip: String = "压缩成ZIP格式"
}
要使用一个属性,只要用名称引用它即可:
fun copyAddress(address: Address): Address {
val result = Address()
result.name = address.name
result.street = address.street
result.city = address.city
result.state = address.state
result.zip = address.zip
return result
}
2、Getters和Setters
声明一个属性的完整语法是:
var <propertyName>[:PropertyType][=<property_initializer>]
[<getter>]
[<setter>]
其初始化器(initalizer)、getter和setter都是可选的。属性类型如果可以从初始化器中推断出来,也可以省略。
//错误:需要显式的初始化器,隐含默认getter和setter
var allByDefault:Int?
//类型Int,默认getter和setter
var initialized = 1
一个只读属性的语法和一个可变的属性的语法有两个方面的不同:1、只读属性用val
代替var
;2、只读属性不允许setter
//类型Int,默认getter必须在构造函数中初始化
val simple:Int?
//类型Int,默认getter
val inferredType = 1
我们可以编写自定义的访问器,非常像普通函数,刚好在属性声明内部。例如:
val isEmpty:Boolean
get() = this.size == 0
//如果可以从getter推断出属性,可以省略
val isEmpty get() = this.size == 0
自定义setter:
var stringPrint: String
get() = this.toString()
set(value) {
print(value)
}
3、幕后字段
Kotlin中类不能有字段。然而,当使用自定义访问器时,有时会有必要拥有一个幕后字段。Kotlin提供一个自动幕后字段,它可以通过使用field
标识符访问。
var counter = 0
set(value) {
if (value >= 0) {
field = value
print(field.toString())
}
}
4、幕后属性
如果求不符合“隐式的幕后字段”,那么可以使用幕后属性:
private var _table: Map<String, Int>? = null
public val table: Map<String, Int>
get() {
if (_table == null) {
_table = HashMap()
}
return _table ?: throw AssertionError("Set ot null by another thread")
}
5、编译期常量
已知值的属性可以使用const
修饰符标记为编译期常量。这些属性需要满足以下要求:
- 位于顶层或者是
object
的一个成员 - 用
String
或原生类型值初始化 - 没有自定义getter
这些属性可以用在注解中:
const val SD:String = "This subsystem is ..."
@Deprecated(SD)fun foo(){}
6、惰性初始化属性
一般地,属性声明为非空类型必须在构造函数中初始化。然而,这经常不方便。例如:属性可以通过依赖注入来初始化,或者在单元测试的setup方法中初始化。这种情况下,不能在构造函数内提供一个非空初始器。但你仍然想在类体中引用该属性是避免空检查。
public class MyTest{
lateinit var subject:TestSubject
@SetUp fun setup(){
subject = TestSubject()
}
@Test fun test(){
subject.method()
}
}
在使用该修饰符时,需满足以下条件:
- 只能用于在类体中声明的var属性
- 该属性没有定义getter或setter时。
- 必须是非空类型,并且不能是原生类型。