
Java中有可见性修饰符(private...),而Kotlin中也是有这样的修饰符,但也有一些不一样,下面来学习Kotlin中的可见性修饰符(Visibility Modifiers)和数据类(Data Classes)的相关知识。
可见性修饰符(Visibility Modifiers)
Kotlin有四种可见性修饰符,分别是public、internal、protected和private,默认的修饰符是public,除了internal之外,其他三种修饰符与Java的访问权限一样。
Kotlin的可见性除了用于类,对象,接口,构造函数,属性之外,还用于属性的setter方法(getter的可见性是由属性的可见性决定的)。
属性可见性修饰
// 文件名: 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 // 在模块中可见
注意:protected在Top-level中不可以使用
private修饰的属性是不会自动生成get和set方法的
在Kotlin中可以使用使用属性名来直接获取值和设置值的,原因就是Kotlin会自动生成属性的get和set方法

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

类和接口的可见性修饰
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关键字。
class C @Inject private constructor(a: Int) { ... }
Kotlin中局部变量,函数和类是不允许使用修饰符的。
模块(Modules)
internal权限是是模块级别的访问权限,可以访问本模块的internal变量和方法,当跨模块的时候就无法访问另一个模块的internal变量或方法。
数据类(Data Classes)
前面说过(Kotlin学习(五): 惯用语法和代码风格),Kotlin的惯用语法是有写POJO类,一般写那种类都是用data修饰的类,也就是数据类表示,只保存数据的类。
data class User(val name: String, val age: Int)
编译器自动从主构造函数中声明的所有属性生成以下方法

可以看出有 getter和setter方法,还有componentN()函数,对应按声明顺序出现的所有属性,如name就是component1(),age就是component2()。
当然还有 equals()、hashCode()、和 toString()(输出的格式为User(name=..., age=...)),为什么生成的class文件中没有这三个类,但是又可以直接调用呢?

Any类是Kotlin每个类的超类,所以自然可以调用这几个方法。
为了确保数据类生成的代码的一致性和有意义,必须满足
- 如果构造函数参数中没有声明是
val或者var,这些函数就不会生成 - 主构造函数需要有至少一个参数
- 数据类不能有
abstract、open、sealed和inner修饰 - 在1.1版本之前,数据类只能实现接口
在构造函数那里也说过,如果生成的类需要一个无参数的构造函数,则必须指定所有属性的默认值
data class User(var name: String = "", var age: Int = 0)
复制(copying)
数据类在创建的时候,除了会生成上面的几个方法外,还会生成一个copy ()函数,copy()能够复制一个对象改变它的一些属性情况下,又要保持其余的不变,如上面的User类,copy()函数的实现:
fun copy(name: String = this.name, age: Int = this.age) = User(name, age)

在使用copy()之后,就可以修改数据类的一些属性了:
val jack = User(name = "jack", age = 1)
val olderJack = jack.copy(age = 2)
Kotlin提供了Pair和Triple作为标准数据类,命名数据类是更好的设计选择。
两个参数的时候使用Pair数据类

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