kotlin 初始化块
前言
使用纯代码 加 注释的方式,可以更快的理解源码
如果你喜欢,请点个赞,后期会不断的深入讲解
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
}
总结
🤩
🎉 原 创 不 易 , 还 希 望 各 位 大 佬 支 持 一 下 \textcolor{blue}{原创不易,还希望各位大佬支持一下} 原创不易,还希望各位大佬支持一下
👍 点 赞 , 你 的 认 可 是 我 创 作 的 动 力 ! \textcolor{green}{点赞,你的认可是我创作的动力!} 点赞,你的认可是我创作的动力!
🌟 收 藏 , 你 的 青 睐 是 我 努 力 的 方 向 ! \textcolor{green}{收藏,你的青睐是我努力的方向!} 收藏,你的青睐是我努力的方向!
✏️ 评 论 , 你 的 意 见 是 我 进 步 的 财 富 ! \textcolor{green}{评论,你的意见是我进步的财富!} 评论,你的意见是我进步的财富!