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 ;

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


3 . 扩展属性初始化问题 : 扩展属性没有初始化器 , 那么就必须处理初始化问题 , 必须定义其 get / set 访问器方法 ;

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


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