[关闭]
@cxm-2016 2016-11-25T16:38:28.000000Z 字数 3030 阅读 4789

Kotlin(八)——属性(Property)与域(Field)

Kotlin

版本:3
翻译:李颖


声明属性

Kotlin 中的类可以拥有属性. 可以使用 var 关键字声明为可变(mutable)属性, 也可以使用 val 关键字声明为只读属性.

  1. public class Address {
  2. public var name: String = ...
  3. public var street: String = ...
  4. public var city: String = ...
  5. public var state: String? = ...
  6. public var zip: String = ...
  7. }

使用属性时, 只需要简单地通过属性名来引用它, 和使用 Java 中的域变量(field)一样:

  1. fun copyAddress(address: Address): Address {
  2. val result = Address() // Kotlin 中没有 'new' 关键字
  3. result.name = address.name // 将会调用属性的访问器方法
  4. result.street = address.street
  5. // ...
  6. return result
  7. }

取值方法(Getter)与设值方法(Setter)


声明属性的完整语法是:

  1. var <propertyName>: <PropertyType> [= <property_initializer>]
  2. [<getter>]
  3. [<setter>]

其中的初始化器(initializer),取值方法(getter),以及设值方法(setter)都是可选的。 如果属性类型可以通过初始化器自动推断得到,或者可以通过这个属性覆盖的基类成员属性推断得到, 则属性类型的声明也可以省略。

示例:

  1. var allByDefault: Int? // 错误: 需要明确指定初始化器, 此处会隐含地使用默认的取值方法和设值方法
  2. var initialized = 1 // 属性类型为 Int, 使用默认的取值方法和设值方法

只读属性声明的完整语法与可变属性有两点不同: 由 val 开头, 而不是 var, 并且不允许指定设置方法:

  1. val simple: Int? // 属性类型为 Int,使用默认的取值方法,属性值必须在构造器中初始化
  2. val inferredType = 1 // 属性类型为 Int,使用默认的取值方法

我们可以编写自定义的访问方法,与普通的函数很类似,访问方法的位置就在属性定义体之内. 下面是一个自定义取值方法的示例:

  1. val isEmpty: Boolean
  2. get() = this.size == 0

自定义设值方法的示例如下:

  1. var stringRepresentation: String
  2. get() = this.toString()
  3. set(value) {
  4. setDataFromString(value) // 解析字符串内容, 并将解析得到的值赋给对应的其他属性
  5. }

Kotlin 的编程惯例是:设值方法的参数名称为 value, 但如果你喜欢, 也可以选择使用不同的名称。

如果你需要改变属性访问方法的可见度,,或者需要对其添加注解,但又不需要修改它的默认实现, 你可以定义这个方法, 但不定义它的实现体:

  1. var setterVisibility: String = "abc"
  2. private set // 设值方法的可见度为 private, 并使用默认实现
  3. var setterWithAnnotation: Any? = null
  4. @Inject set // 对设值方法添加 Inject 注解

属性的隐藏域变量(Backing Field)

Kotlin 的类不能拥有域变量。但是,使用属性的自定义访问器时,有时会需要隐藏域变量(backing field). 为了这种目的, Kotlin 提供了一种自动的隐藏域变量, 可以通过 field 标识符来访问:

  1. var counter = 0 // 初始化给定的值将直接写入后端域变量中
  2. set(value) {
  3. if (value >= 0)
  4. field = value
  5. }

field 标识符只允许在属性的访问器函数内使用。

如果属性 get/set 方法中的任何一个使用了默认实现, 或者在 get/set 方法的自定义实现中通过 field 标识符访问属性, 那么编译器就会为属性自动生成隐藏域变量.

比如,下面的情况不会存在隐藏域变量:

  1. val isEmpty: Boolean
  2. get() = this.size == 0

隐藏属性(Backing Property)

如果你希望实现的功能无法通过这种 “隐含的域变量” 方案来解决, 你可以使用 隐藏属性(backing property) 作为替代方案:

  1. private var _table: Map<String, Int>? = null
  2. public val table: Map<String, Int>
  3. get() {
  4. if (_table == null)
  5. _table = HashMap() // 类型参数可以自动推断得到, 不必指定
  6. return _table ?: throw AssertionError("Set to null by another thread")
  7. }

不管从哪方面看, 这种方案都与 Java 中完全相同, 因为隐藏私有属性的取值方法与设值方法都使用默认实现, 我们对这个属性的访问将被编译器优化, 变为直接读写隐藏域变量, 因此不会发生不必要的函数调用, 导致性能损失。

编译期常数值


如果属性值在编译期间就能确定, 则可以使用 const 修饰符, 将属性标记为编译期常数值(compile time constants)。 这类属性必须满足以下所有条件:

这类属性可以用在注解内:

  1. const val SUBSYSTEM_DEPRECATED: String = "This subsystem is deprecated"
  2. @Deprecated(SUBSYSTEM_DEPRECATED) fun foo() { ... }

延迟初始化属性(Late-Initialized Property)


通常, 如果属性声明为非 null 数据类型, 那么属性值必须在构造器内初始化. 但是, 这种限制很多时候会带来一些不便. 比如, 属性值可以通过依赖注入来进行初始化, 或者在单元测试代码的 setup 方法中初始化. 这种情况下, 你就无法在构造器中为属性编写一段非 null 值的初始化代码, 但你仍然希望在类内参照这个属性时能够避免 null 值检查。

要解决这个问题, 你可以为属性添加一个 lateinit 修饰符:

  1. public class MyTest {
  2. lateinit var subject: TestSubject
  3. @SetUp fun setup() {
  4. subject = TestSubject()
  5. }
  6. @Test fun test() {
  7. subject.method() // 直接访问属性
  8. }
  9. }

这个修饰符只能用于 var 属性, 而且只能是声明在类主体部分之内的属性(不可以是主构造器中声明的属性), 而且属性不能有自定义的取值方法和设值方法。 属性类型必须是非 null 的,,而且不能是基本类型。

在一个 lateinit 属性被初始化之前访问它, 会抛出一个特别的异常, 这个异常将会指明被访问的属性, 以及它没有被初始化这一错误。

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注