kotlin 基础 初始化块
kotlin 基础入门
第五章 kotlin 初始化块
文章目录
- kotlin 基础入门
- 前言
-
- 1、什么是初始化块 init 和主、 次构造函数
- 2、构造函授初始化顺序
- 3、延迟初始化 lateinit
- 4、惰性初始化 by lazy
- 5、初始化陷阱 01
- 6、初始化陷阱 02
- 7、初始化陷阱 03
前言
使用纯代码 加 注释的方式,可以更快的理解源码
如果你喜欢,请点个赞,后期会不断的深入讲解
1、什么是初始化块 init 和主、 次构造函数
User(name = "张三", age = 21, sex = '女') // 这里会调用主构造函授 User(name = "张三", age = 21, sex = '女', height = 180) // 这里会调用次构造函授 // user 属性是主构造函数class User(name: String, age: Int, sex: Char) { // 这个不是Java 中的 static {}// 相当于是Java的 {} 构造代码块// 在kotlin 中, 是初始化块, init 构造代码块 init { println("主构造函数被调用了 $name, $age, $sex") } // constructor 是主构造函数 user 的 次构造函数 constructor(name: String, age: Int, sex: Char, height: Int) : this("tiger", 18, '男'){ println("次构造函数被调用了") }}
2、构造函授初始化顺序
User(name = "张三", age = 21, sex = '女', height = 180)// 第一步:调用主构造函数class User(_name: String, age: Int, sex: Char) {// 第二步: 生成val nName (其实这个成员变量和 init 是同时生成的,只是 val nName = _name 写在了init的前面,所以才先执行) val nName = _name; init { val nameValue = nName; // 第三步: 生成 nameValue 的细节 println("init 代码块打印: $nameValue") } constructor(name: String, age: Int, sex: Char, height: Int) : this("tiger", 18, '男'){// 第四步: 生成次构造的细节 println("次构造函数被调用了") }}
3、延迟初始化 lateinit
lateinit 在使用的时候,需要手动加载的懒加载
val lateManager = LateManager() lateManager.loadRequest() lateManager.showResponseResult()class LateManager{// lateinit val AAA; val 是不可修改函数,在后面都无法修改了,肯定不能用val lateinit var responseResultInfo: String // lateinit 懒加载模式,在使用的时候,才会被加载// 模拟服务器加载 fun loadRequest(){ responseResultInfo = "服务器加载成功!" } fun showResponseResult(){// 由于responseResultInfo 没有初始化,一调用就会直接导致崩溃,这里我们使用一个判断,判断 responseResultInfo 是否初始化 if (::responseResultInfo.isInitialized){ println("responseResultInfo: $responseResultInfo") }else{ println("你都还么有初始化,是不是忘了!") } }}
4、惰性初始化 by lazy
惰性初始化 by lazy 是在使用的时候,自动加载的 懒加载方式
val serverData = ServerData()// 模拟数据库的加载,写一个定时器 Thread.sleep(5000) println("开始加载数据") println(serverData.dataBase)class ServerData{ val dataBase by lazy { readSqlServerDatabaseAction() } private fun readSqlServerDatabaseAction(): String { println("数据加载中,请等待。。。。。。") println("数据加载中,请等待。。。。。。") println("数据加载中,请等待。。。。。。") println("数据加载中,请等待。。。。。。") println("数据加载中,请等待。。。。。。") println("数据加载中,请等待。。。。。。") println("数据加载中,请等待。。。。。。") println("数据加载中,请等待。。。。。。") return "database data load success ok"; }}
5、初始化陷阱 01
class ServerData{ // 在这里,看上去,代码没有任何问题,但是我们前面讲了,成员变量 number 和 init 是同级的。// 那么就意味着 init 执行的时候,成员变量 number 还没有初始化// 这里不要套入Java的思想,而是以顺序的思想去考虑 init { number = number.times(9) // times(9) 指 number * 9 } var number = 9}
6、初始化陷阱 02
class ServerData{// 初始化的顺序 ServerData -> var info: String -> init -> getInfoMethod() -> info = "tiger"// 这样会直接造成info 返回的是一个null ,因为在进入 init 之后,直接调用的就是getInfoMethod()方法,这时候 成员变量 info 还没有进行初始化赋值 var info: String init { getInfoMethod() info = "tiger" } private fun getInfoMethod() { println("info: $info") }}
7、初始化陷阱 03
// 这样写,看着没有什么问题,但是结合函数里面的代码,会直接导致崩溃// 当你调用 主构造函数,传入info, 再执行成员变量,调用getInfoMethod() 函数,获取字符长度的时候,其实你获取的是null println("内容的字符长度是:${User("tiger").content.length}")class User(_info: String) { val content = getInfoMethod() private val info = _info private fun getInfoMethod() = info}