Kotlin学习_可见性修饰符和数据类

Interoperable

Java中有可见性修饰符(private…),而Kotlin中也是有这样的修饰符,但也有一些不一样,下面来学习Kotlin中的可见性修饰符(Visibility Modifiers)和数据类(Data Classes)的相关知识。

可见性修饰符(Visibility Modifiers)

Kotlin有四种可见性修饰符,分别是publicinternalprotectedprivate,默认的修饰符是public,除了internal之外,其他三种修饰符与Java的访问权限一样。

Kotlin的可见性除了用于类,对象,接口,构造函数,属性之外,还用于属性的setter方法(getter的可见性是由属性的可见性决定的)。

属性可见性修饰

1
2
3
4
5
6
7
8
9
10
// 文件名: example.kt
package foo

private fun foo() {} // 在example.kt中可见

public var bar: Int = 5 // 任何地方都可见
// 因为get的可见性是由属性决定的,所有这里的get的可见性为public
private set // setter在example.kt中可见

internal val baz = 6 // 在模块中可见

注意:protectedTop-level中不可以使用

private修饰的属性是不会自动生成get和set方法的

在Kotlin中可以使用使用属性名来直接获取值和设置值的,原因就是Kotlin会自动生成属性的getset方法

但是,如果如果设置属性的可见性是private的话,我们知道private修饰的是私有的,也就是当前类可用,那么就不会自动生成getset方法了,不能给外部访问。

类和接口的可见性修饰

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
open class Outer {
private val a = 1 // 在Outer类中可见
protected open val b = 2 // 在Outer类和子类中可见
internal val c = 3 // 在模块中可见
val d = 4 // 默认为public

protected class Nested {
public val e: Int = 5
}
}

class Subclass : Outer() {
// a不可见
// b,c和d可见
// Nested类和e可见

override val b = 5 // 重写后的b依然是protected
}

class Unrelated(o: Outer) {
// o.a和o.b不可见
// o.d可见,在相同模块下o.c可见
// Outer.Nested不可见, Nested::e也是不可见
}

注意:外部类不可以访问内部类的private成员。

构造函数同样可以设置可见性,前面说过,在设置主构造函数的可见性或者注解的时候,要加上constructor关键字。

1
class C @Inject private constructor(a: Int) { ... }

Kotlin中局部变量,函数和类是不允许使用修饰符的。

模块(Modules)

internal权限是是模块级别的访问权限,可以访问本模块的internal变量和方法,当跨模块的时候就无法访问另一个模块的internal变量或方法。

数据类(Data Classes)

前面说过(Kotlin学习(五): 惯用语法和代码风格),Kotlin的惯用语法是有写POJO类,一般写那种类都是用data修饰的类,也就是数据类表示,只保存数据的类。

1
data class User(val name: String, val age: Int)

编译器自动从主构造函数中声明的所有属性生成以下方法

User类

可以看出有 gettersetter方法,还有componentN()函数,对应按声明顺序出现的所有属性,如name就是component1()age就是component2()

当然还有 equals()hashCode()、和 toString()(输出的格式为User(name=..., age=...)),为什么生成的class文件中没有这三个类,但是又可以直接调用呢?

Any类

Any类是Kotlin每个类的超类,所以自然可以调用这几个方法。

为了确保数据类生成的代码的一致性和有意义,必须满足

  • 如果构造函数参数中没有声明是val或者var,这些函数就不会生成
  • 主构造函数需要有至少一个参数
  • 数据类不能有abstractopensealedinner修饰
  • 在1.1版本之前,数据类只能实现接口

构造函数那里也说过,如果生成的类需要一个无参数的构造函数,则必须指定所有属性的默认值

1
data class User(var name: String = "", var age: Int = 0)

复制(copying)

数据类在创建的时候,除了会生成上面的几个方法外,还会生成一个copy ()函数,copy()能够复制一个对象改变它的一些属性情况下,又要保持其余的不变,如上面的User类,copy()函数的实现:

1
fun copy(name: String = this.name, age: Int = this.age) = User(name, age)

copy

在使用copy()之后,就可以修改数据类的一些属性了:

1
2
val jack = User(name = "jack", age = 1)
val olderJack = jack.copy(age = 2)

Kotlin提供了PairTriple作为标准数据类,命名数据类是更好的设计选择。
两个参数的时候使用Pair数据类
Pair

三个参数的时候使用Triple数据类
Triple