Android - 一些Kotlin写法

本文介绍了Kotlin中的延迟初始化技术,包括`lateinit`和`by lazy`的使用场景和优势。`lateinit`用于非null变量的延迟初始化,避免了频繁的非空检查。`by lazy`提供线程安全的懒加载,适用于不立即初始化的对象。此外,还讨论了使用密封类优化`when`表达式,Json字段映射,以及全局Context的获取。最后,展示了ViewModel的正确初始化方式。

一、全局变量(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:

  1. 创建一个 APP 类继承自 Application。
  2. 伴生对象中定义 context 属性,重写 onCreate() 函数,调用 getApplicationContext() 将获取到的上下文赋值给 context。
  3. Activity、Service这样的 Context 设置成静态容易造成内存泄漏,但我们这里获取的是 Application 中的 Context,全局只有一份切在整个APP生命周期内不会被回收因此不存在该风险,使用注解忽略 IDE 警告。
  4. 一定要在 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()) }

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值