【Kotlin】扩展属性 ( 扩展变量属性 | 扩展常量属性 | 注意事项 | 本质分析 )

文章目录

I . 扩展属性 总结

II . 扩展属性 定义格式

III . 扩展属性 标准示例

IV . 扩展属性 注意事项

V . 扩展属性 本质分析

VI . 扩展属性 代码示例解析



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


上一篇:Jquery和JS的区别


下一篇:我们为什么要迁移PHP到HHVM