欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 汽车 > 时评 > kotlin与MVVM结合使用总结(一)

kotlin与MVVM结合使用总结(一)

2025/4/24 22:03:13 来源:https://blog.csdn.net/2301_80329517/article/details/146187631  浏览:    关键词:kotlin与MVVM结合使用总结(一)

一、Kotlin 与 MVVM 结合的核心优势

  1. 代码简洁性

    • 数据类(data class)简化 Model 层定义,自动生成equals/hashCode/toString
    • 扩展函数简化 View 层逻辑(如点击事件扩展)
    • lateinit/by lazy优化 ViewModel 属性初始化
  2. 异步处理优化

    • 协程(Coroutines)替代 RxJava,轻量且代码可读性强
    • withContext(Dispatchers.IO)切换线程,配合LiveData更新 UI
  3. 响应式编程

    • LiveData + StateFlow实现数据双向绑定
    • Flow替代LiveData处理复杂数据流(如网络请求重试)
  4. 生命周期感知

    • ViewModel配合SavedStateHandle保存状态
    • LifecycleOwner简化生命周期监听

二、MVVM 实现细节

  1. ViewModel 层

    • 使用 Kotlin @HiltViewModel注解依赖注入(结合 Hilt)
    • 协程启动任务:viewModelScope.launch { ... }
    • StateFlow封装业务状态,替代可变 LiveData
  2. View 层

    • DataBinding 绑定布局,Kotlin 表达式简化逻辑(如@{user.name ?: "Guest"}
    • ViewBinding替代findViewById,类型安全
    • 协程与lifecycleScope结合处理异步任务
  3. Model 层

    • 数据类定义实体,@SerializedName配合 Retrofit 解析 JSON
    • 仓库(Repository)模式隔离数据源,Kotlin 密封类定义请求状态(如Result.Success/Error

之间的关联:

  • View 持有 ViewModel:View(如 Activity、Fragment 等)会创建并持有 ViewModel 的引用,通过数据绑定机制观察 ViewModel 中的数据变化。
  • ViewModel 持有 Model:ViewModel 持有 Model 的引用,从 Model 获取数据并处理业务逻辑,将处理后的数据暴露给 View。
  • Model 不持有 View 和 ViewModel:Model 专注于数据的存储和获取,不依赖于 View 和 ViewModel。

三、面试高频问题

1、ViewModel 是如何保持数据的

        ViewModel 使用了 Android 架构组件中的 SavedStateHandle 来保持数据。

        SavedStateHandle 是一个键值对集合,用于在配置更改(如屏幕旋转)时保存和恢复数据。

       当 Activity 或 Fragment 因配置更改而销毁重建时,ViewModel 不会被销毁,SavedStateHandle 中的数据会被保留,从而实现数据的保持。

2、 ViewModel 是怎么做到在 Activity 销毁重建新实例之后还能保持不变的

        在 Android 中,ViewModel 的生命周期与 Activity 或 Fragment 的生命周期不同。

        当 Activity 或 Fragment 因配置更改(如屏幕旋转)而销毁重建时,系统会自动保留 ViewModel 的实例。

       这是通过 ViewModelStore 来实现的,ViewModelStore 是一个存储 ViewModel 实例的容器,Activity 或 Fragment 会持有一个 ViewModelStore 的引用。

      当 Activity 或 Fragment 重建时,会从 ViewModelStore 中获取之前的 ViewModel 实例,从而保证 ViewModel 中的数据不会丢失。

四、最佳实践

  • View:对应 Android 中的 Activity、Fragment、View 等,负责界面的绘制和用户交互的处理。
  • ViewModel:对应 Android 中的 ViewModel 类,负责处理业务逻辑和数据的转换,通过 LiveData、StateFlow 等将数据暴露给 View。
  • Model:对应数据的实体类(如 Kotlin 中的数据类)和数据获取的仓库类(Repository),负责数据的存储和获取。

演示代码:

ViewModel:

@HiltViewModel
class MainViewModel @Inject constructor(private val repository: UserRepository
) : ViewModel() {private val _user = MutableStateFlow<User?>(null)val user: StateFlow<User?> = _userfun fetchUser(userId: String) {viewModelScope.launch {try {_user.value = repository.getUser(userId)} catch (e: Exception) {// 处理错误}}}
}

View 层(Fragment):

class MainFragment : Fragment() {private val viewModel by viewModels<MainViewModel>()override fun onViewCreated(view: View, savedInstanceState: Bundle?) {super.onViewCreated(view, savedInstanceState)viewModel.user.collectAsState().observe(viewLifecycleOwner) { user ->// 更新UI}}
}

真实操作:

首先,确保在项目的 build.gradle 中添加必要的依赖,如 ViewModel、LiveData、Kotlin 协程等:

dependencies {// ViewModel 和 LiveDataimplementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.6.2'implementation 'androidx.lifecycle:lifecycle-livedata-ktx:2.6.2'// Kotlin 协程implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.7.3'
}

Model 层

Model 层主要负责数据的定义和数据的获取。这里以一个简单的用户数据为例:

// 定义用户数据类
data class User(val id: Int, val name: String, val age: Int)// 模拟数据获取的仓库类
class UserRepository {// 模拟网络请求,使用协程进行异步操作suspend fun getUser(id: Int): User {// 模拟耗时操作delay(1000)return User(id, "John Doe", 30)}
}

ViewModel 层

ViewModel 层负责处理业务逻辑,并将数据暴露给 View 层。它通过 LiveData 或 StateFlow 来实现数据的响应式更新。

import androidx.lifecycle.ViewModel
import androidx.lifecycle.viewModelScope
import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.launchclass UserViewModel(private val userRepository: UserRepository) : ViewModel() {// 使用 MutableStateFlow 来存储和更新用户数据private val _user = MutableStateFlow<User?>(null)// 对外暴露不可变的 StateFlowval user: StateFlow<User?> = _user// 获取用户数据的方法fun fetchUser(id: Int) {viewModelScope.launch {try {// 调用仓库类的方法获取用户数据val user = userRepository.getUser(id)// 更新 StateFlow 的值_user.value = user} catch (e: Exception) {// 处理异常e.printStackTrace()}}}
}

View 层

View 层通常是 Activity 或 Fragment,负责显示数据和处理用户交互。这里以 Fragment 为例:

import android.os.Bundle
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import androidx.fragment.app.Fragment
import androidx.lifecycle.lifecycleScope
import kotlinx.coroutines.flow.collect
import kotlinx.coroutines.launchclass UserFragment : Fragment() {private val userViewModel: UserViewModel by lazy {UserViewModel(UserRepository())}override fun onCreateView(inflater: LayoutInflater,container: ViewGroup?,savedInstanceState: Bundle?): View? {return inflater.inflate(R.layout.fragment_user, container, false)}override fun onViewCreated(view: View, savedInstanceState: Bundle?) {super.onViewCreated(view, savedInstanceState)// 启动协程来收集 StateFlow 的数据lifecycleScope.launch {userViewModel.user.collect { user ->user?.let {// 更新 UI// 这里可以根据实际情况更新 TextView 等视图组件// 例如:textView.text = "${it.name}, ${it.age}"}}}// 触发数据获取userViewModel.fetchUser(1)}
}

代码解释

  • Model 层
    • data class User:使用 Kotlin 的数据类简洁地定义了用户数据结构,自动生成 equalshashCode 和 toString 方法。
    • UserRepository:模拟了数据的获取过程,使用 suspend 关键字和 delay 函数模拟网络请求的异步操作,使用协程进行异步处理。
  • ViewModel 层
    • MutableStateFlow 和 StateFlow:用于存储和暴露用户数据,实现数据的响应式更新。MutableStateFlow 用于内部数据的更新,StateFlow 用于对外暴露不可变的数据。
    • viewModelScope.launch:在 ViewModel 中使用协程进行异步操作,确保在 ViewModel 的生命周期内执行。当 ViewModel 被销毁时,协程会自动取消。
  • View 层
    • lifecycleScope.launch:在 Fragment 中使用协程来收集 StateFlow 的数据,确保在 Fragment 的生命周期内执行。当 Fragment 被销毁时,协程会自动取消。
    • userViewModel.fetchUser(1):触发数据获取操作,调用 ViewModel 中的方法获取用户数据。

总结

     Kotlin 通过协程、数据类、扩展函数等特性大幅提升了 MVVM 的开发效率和代码质量,

    面试中需重点关注异步处理、数据绑定、依赖注入及生命周期管理。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词