I . 扩展属性 总结
扩展属性 总结 :
① 修饰 : 变量扩展属性使用 var 修饰 , 常量扩展属性使用 val 修饰 ;
② 属性访问器定义 : 变量属性必须定义 get 和 set 属性访问器方法 , 常量属性必须定义 get 属性访问器方法 ;
③ 注意事项 : 扩展属性没有幕后字段 , 没有初始化器 , 必须定义属性访问器 ;
④ 扩展属性本质 : 扩展属性没有在内存中分配字段保存该属性 , 其本质是一个经过计算得出的值 , 扩展变量属性相当于扩展了两个属性访问器方法 ;
II . 扩展属性 定义格式
1 . 扩展变量属性定义格式 : 扩展的变量属性 , 使用 var 修饰属性 , 必须定义 get / set 属性访问器方法 ;
var 接收者类型.扩展属性名称 : 扩展属性类型
get(){}
set(value){}
2 . 扩展常量属性定义格式 : 扩展的变量属性 , 使用 val 修饰 , 必须定义 get 方法 , 必须不能定义 set 方法 ;
val 接收者类型.扩展属性名称 : 扩展属性类型
get(){}
III . 扩展属性 标准示例
扩展属性要素 :
1 . 修饰符 : 使用 var 修饰扩展的变量属性 , 使用 val 修饰扩展的常量属性 ;
2 . 属性访问器定义 : 变量必须定义 getter 和 setter 属性访问器 , 常量扩展属性只能且必须定义 getter 方法 ;
open class Student {
var name : String = "Tom"
var age : Int = 18
}
//扩展变量属性
Student.olderAge : Int
get() {
return this.age + 1
}
set(value) {
this.age = value - 1
}
//扩展常量属性
val Student.nameAndage : String
get() {
return "${this.name} : ${this.age}"
}
IV . 扩展属性 注意事项
1 . 没有幕后字段 : 不管是扩展函数 , 还是扩展属性 , 都没有将扩展的成员加入到类中 , 扩展属性没有 幕后字段 ;
2 . 不能有属性初始化器 : 扩展属性由于没有幕后字段 , 因此不能定义属性的初始化器 , 给扩展属性定义初始化器有如下报错信息 Extension property cannot be initialized because it has no backing field
;
3 . 扩展属性初始化问题 : 扩展属性没有初始化器 , 那么就必须处理初始化问题 , 必须定义其 get / set 访问器方法 ;
4. 扩展属性不能保存值 : 扩展属性没有幕后字段不能保存实际的字段值 , 其属性访问器中只能调用对象中的属性和方法 , 不能调用扩展属性本身 , 会栈溢出 ;
V . 扩展属性 本质分析
1 . 没有实际字段字段 : 扩展属性并不是一个实际的属性 , 没有幕后字段 , 不能存储一个新的字段值 ;
2 . 扩展属性可访问内容 : 扩展属性的属性访问器中只能调用接收者类型对象中的其它成员 , 不能调用扩展属性本身 ;
3 . 本质是扩展函数 : 扩展属性本质相当于定义了属性访问器方法 , 在该方法中可以对扩展的接收者类对象进行各种操作 , 其本质是扩展了两个函数 ( get / set 属性访问器 ) ;
VI . 扩展属性 代码示例解析
扩展属性代码示例解析 :
1 . olderAge 扩展属性 : 该属性是 var 修饰的扩展的变量属性 , 必须定义该属性的 get 和 set 属性访问器 ;
2 . olderAge 扩展属性的本质 :
① 没有为 olderAge 属性分配内存 : 在内存中是没有准备任何字段来存储这个值的 , 因此实际上在内存中是没有为该值分配内存空间 ;
② 属性访问器可访问内容 : 只能通过调用 Student 接收者类型对象中的成员属性 , 成员方法 , 或定义局部变量 ;
③ 扩展属性本质 : 使用上述可访问内容 , 经过计算模拟生成一个属性 , 这里模拟的属性就是比 Student 对象的 age 大一岁的 年龄属性 ;
3 . nameAndage 扩展属性 : 是一个常量 , 使用 val 修饰 , 因此其只能定义 get 属性访问方法 , 其本质与上面的变量本质相同 , 内存中没有实际值 , 是一个经过运算生成的值 ;
4 . 扩展属性 代码示例 :
open class Student {
var name : String = "Tom"
var age : Int = 18
}
/*
扩展属性并不是添加了一个属性
没有额外扩展一个存储任何值的字段
可以利用原来的类模拟生成一个字段
本质是 : 为该接收者类扩展了两个方法
该扩展的属性意义是 , 定义了一个属性 , 比类中的年龄大一岁
*/
var Student.olderAge : Int
get() {
//扩展属性的访问器中 , 可以访问任何接收者类对象中的属性和方法
// 绝对不能访问该属性本身 , 会造成递归溢出
return this.age + 1
}
set(value) {
this.age = value - 1
}
//扩展常量属性
val Student.nameAndage : String
get() {
return "${this.name} : ${this.age}"
}
fun main() {
var student : Student = Student()
student.olderAge = 18
//18
println(student.olderAge)
//Tom : 17
println(student.nameAndage)
}
5 . 执行结果 :
18
Tom : 17