简介
本篇博客主要是介绍Kotlin语法中的【类的属性和字段】相关的知识,帮助各位更快的掌握Kotlin,如果有疏漏或者错误,请在留言中指正,谢谢。 系列汇总地址
基础知识
定义
在讲解之前我感觉有很多人根本不清楚什么是属性,什么又是字段,我们先简单说明一下。
简单理解:
字段,通常叫做类成员或者类成员变量,理解为"数据成员",用来承载数据。
属性,通常可以理解为set和get方法。其属名性时根据get和set方法名得出的,规则是:去掉get或set后其剩余的字符串,属性大多是对字段的封装,限制其访问和写入。
具体说明(以java为例):
- 字段,在类中定义的成员变量。
public class A{
private String s = "123";
}
我们可以描述为A类中有个s字段
- 属性,只局限于类中方法的声明,并不与类中其他成员相关。
void setA(String s){}
String getA(){return s}
当一个类中拥有这样一对方法时,我们可以说,这个类中拥有一个可读写的a属性(注意是小写a)。如果去掉了set
的方法,则是可读属性,反之亦然。
总结一波:通过上面的介绍我们了解了什么是属性以及什么是字段,在java
和Kotlin
中其含义是一致的。
属性和字段
声明属性
对于Kotlin的类可以有属性。我们之前也说过可以通过var
声明可变属性,val
声明只读属性,或许你会问了,为什么不是叫字段而是叫属性呢?因为Kotlin
类中声明的变量,都会提供默认的get
、set
(val没有该方法)方法,所以声明的都是属性。Kotlin
中没有字段,只有幕后字段(backing filed)。
我们声明的时候大都采用下面的形式:
class People {
var name: String? = null
var address: String = "地球"
var age: Int? = null
}
你或许会说,属性是有get、set方法的(不仅限于此),那Kotlin自己实现的怎么用呢?
var people=People() //声明实例,不需要new
people.name //直接使用,内部会调用访问方法
上面我们看到使用起来还是比较简单的。类似于我们java
中声明了私有变量,然后提供了set、get
方法,到达属性的目的。
Getters和Setters
声明一个属性的完整语法如下:
【属性修饰符(var、val)】+【属性名称】+ 【:】 + 【类型】 = 【初始化】
【gettter】
【setter】
对于【初始化】和【getter】和【setter】都是可选的,如果通过【初始化】的值或者【getter】可以推断出类型,则【类型】也可以省略。
属性通过var
和val
去声明得到只读属性和可变属性,那他们的区别究竟是什么呢?
只读属性和可变属性的区别:
只读属性使用
val
声明,可变属性使用var
声明只读属性不允许
setter
看到此处你或许会感叹Kotlin
的厉害之处,省了不知多少时间,但有些人或许会感叹了,如果我想自己定义属性的访问方法又怎么弄呢?像java
再自定义方法吗?会不会和默认的setter
或者getter
冲突呢?对于疑问我们一一解答。
自定义Getters和Setters
编写Getters
和Setters
非常像一般的不同方法,在属性声明内部,举例:
Getters方法:
var name: String? = null //声明name属性
get() {//重写get方法
return "hhh"
}
var people=People()
people.name
此处的name
的值便是hhh
,且无论name
设置成什么值。
Setters方法:
set(value) {//错误写法
name=value //1)
}
在此不知道有没有人看出不对的地方,上面 1) 处会导致循环调用,为什么呢?当你对属性赋值的时候就会调用set方法,当你获取属性的值得时候就会使用get方法。所以上面会一直循环调用set方法。
下面我们看看正确的怎么写:
set(value) {
field=value
}
看完上面的代码,你应该有两个疑问,1.value 是什么鬼,field 又是什么鬼。。。
好,我们说明一下:
value
是setter
的参数,其类型同于属性的类型,不爽你也可以换个其他名字。。field
就是我们之前说的幕后字段,用于将真正的值赋值给属性,而不会导致循环调用。它只能在属性的访问器中使用。这个也是可选项,有的时候必须(比如上面的Setter),有的时候不是必须(比如上面的Getter)。
如果上面的幕后字段仍然不能满足你...还有更猛的,幕后属性。
private var _table: HashMap<String, String>? = null//私有属性
var table: HashMap<String, String>? = null
get() {
if (_table == null) {
_table = HashMap<String, String>()
}
return _table //返回私有属性
}
上面的代码需要特殊说明一下,当属性被定义为private
后其Getter
和Setter
,都是私有的,外部都不可以访问也就是说 person._table
是不允许的。也就进行了隐藏。又因为默认的Setter
和Getter
调用私有属性会被进行优化,所以不会引入函数调用开销。
编译期常量
在kotlin
中已知值得属性可以使用const
标记为编译期常量。
它需要满足以下条件:
- 使用
val
声明的常量 - 位于顶层或者
object
的一个成员 - 用
String
或者原生类型值初始化 - 没有自定义
Getter
延迟初始化属性
对于非空属性,在声明时必须对其进行初始化,如果想进行延迟初始化,可以使用lateinit
标记,代码举例:
lateinit var name: String
需要注意以下几点:
- 该修饰符只能用于类体中不是方法中的属性
- 没有自定义
Setter
或者Getter
- 不能是原生类型(String可以,它不是原生类型)
如果在初始前访问lateinit
定义的属性会抛出特定异常,指明该属性没有被初始化。
总结一波:我们需要区分Kotlin中和java中使用属性的区别,对于Koltin
中使用【类名.属性】等同于java中的【类名.属性对应方法】,以及牢记Setter
的写法,别写错了,导致循环调用,熟练使用幕后字段。重写后的Getter
和Setter
会覆盖之前的默认方法并不会冲突
总结
至此已经学完了Kotlin的【类的属性和字段】的知识,多回顾多思考,继续后续内容。