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
数据类