一、全局变量(lateinit)
我们经常会抽取一些全局变量方便类中局部调用(view,adapter等),但对属性设置默认为 null 的话,在调用的时候不得不手写大量的非空调用来满足 Kotlin 的空指针检查,即便你明确它不会为null(在 onCreate() 中有初始化过)。
这个时候使用延迟初始化关键字lateinit ,告诉编译器我们会在晚些时候进行初始化,就不用一开始赋值为null。但是一定要确保调用前有被初始化,否则报错UninitializedPropertyAccessException,可以使用代码【::变量名.isInitalized】先做判断是否初始化过。
UI {
private lateinit var adapter: MyAdapter
override fun onCreate(savedInstanceState: Bundle?) {
//直接初始化
adapter = MyAdapter(list)
//判断再赋值,避免重复初始化
if(!::adapter.isInitialized) { adapter = MyAdapter(list) }
}
override fun onClick(v: View?) {
adapter.notifyDataSetChanged()
}
}
二、延迟初始化(by lazy)
- 使用场景:当某个对象或属性不必在类加载的时候就初始化,而是等到调用它的时候再初始化,线程安全。
- 需要注意:与生命周期相关的方法都是在主线程,在其中调用的东西会立即初始化和线程安全,因此没必要使用。
三、ViewHolder(密封类)
由于编译器知道一共有几种类型,可用来优化 when 对分支的判断:
- 后期新增holder,译器会提醒未穷举不给通过,不用担心忘记修改 when 判断最后走了 else 默认情况。
- 很明确只存在几种holder,却为了满足编译器的要求写 else 无效的默认分支。
①新建一个 MsgHolder.kt,编写各种holder。
//定义一个密封类继承自RecyclerView.ViewHolder
sealed class MsgHolder(view: View) : RecyclerView.ViewHolder(view)
//让实际用到的holder去继承它
class LeftHolder(view: View) : MsgHolder(view) {
val tv: TextView = view.findViewById(R.id.tv_left)
}
class RightHolder(view: View) : MsgHolder(view) {
val tv: TextView = view.findViewById(R.id.tv_right)
}
//将 Adapter 的泛型指定为我们定义的 MsgHolder
class MsgAdapter(val list: List<Mag>) : RecyclerView.Adapter<MsgHolder>() {
override fun onBindViewHolder(holder: MsgHolder, position: Int) {
//判断的时候就能不写else分支
when (holder) {
is LeftHolder -> ...
is RightHolder -> ...
}
}
}
四、Json字段关联不同名称的属性(Data Class)
//Json
"formatted_address":"中国北京市"
//Data Class
data class Place(@SerializedName("formatted_address") val address: String)
五、获取全局Context(Application)
将自定义属性 context 改成其它名称如 appContext,就不会报提醒,不用注解忽略。
由于在一些地方无法调用API获取Context:
- 创建一个 APP 类继承自 Application。
- 伴生对象中定义 context 属性,重写 onCreate() 函数,调用 getApplicationContext() 将获取到的上下文赋值给 context。
- Activity、Service这样的 Context 设置成静态容易造成内存泄漏,但我们这里获取的是 Application 中的 Context,全局只有一份切在整个APP生命周期内不会被回收因此不存在该风险,使用注解忽略 IDE 警告。
- 一定要在 Manifest 中配置我们自定义的!
class APP : Application() {
companion object {
@SuppressLint("StaticFieldLeak") //注解忽略警告
lateinit var context: Context //写在伴生对象里才能静态调用,但这里无法调用获取函数所以用lateinit
private set //收窄权限
}
override fun onCreate() {
super.onCreate()
context = applicationContext //在APP创建的时候就赋值
}
}
//别忘了在Manifest中配置
<application
android:name=".APP">
</application>
六、ViewModel初始化
//不用丢到 by lazy 中
val viewModel by lazy { ViewModelProvider(this).get(MainViewModel::class.java) }
//而是这样
val viewModel by viewModels<MainViewModel>()
//有参的
val viewModel by viewModels<MainViewModel> { MainViewModelFactory(MainRepository()) }