kotlin中的模块化结构组件及工作原理

2025-03-15 13:50

本文主要是介绍kotlin中的模块化结构组件及工作原理,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《kotlin中的模块化结构组件及工作原理》本文介绍了Kotlin中模块化结构组件,包括ViewModel、LiveData、Room和Navigation的工作原理和基础使用,本文通过实例代码给大家...

模块化结构组件包含ViewModel、LiveData、Room 和 Navigation ,我将讲解它们的工作原理和基础使用。

ViewModel 工作原理

  • 创建与存储机制:当调用 ViewModelProvider 的 get 方法获取 ViewModel 实例时,ViewModelProvider 会先检查 ViewModelStore 中是否已存在该类型的实例。若存在则直接返回,若不存在则使用 ViewModelProvider.Factory 创建新实例并存储在 ViewModelStore 中。每个 Activity 和 Fragment 都有各自对应的 ViewModelStore,用于管理其内部的 ViewModel 实例。
  • 生命周期管理ViewModel 的生命周期与关联的 Activity 或 Fragment 紧密相关,但又有区别。在配置更改(如屏幕旋转)时,Activity 或 Fragment 会重新创建,而 ViewModelStore China编程会被保留,所以 ViewModel 实例也得以保留,从而保证数据的一致性。当 Activity 或 Fragment 被销毁(非因配置更改)时,ViewModelStore 会调用 clear 方法,进而调用 ViewModel 的 onCleared 方法,让开发者可以在此进行资源释放操作。

ViewModel 通过 ViewModelStore 存储实例,在配置更改时保留数据,在关联组件非配置更改销毁时释放资源。 

// 定义 ViewModel 类
import androidx.lifecycle.ViewModel
import androidx.lifecycle.MutableLiveData
class NewsViewModel : ViewModel() {
    // 定义 LiveData 存储新闻列表
    private val _newsList = MutableLiveData<List<String>>()
    val newsList: LiveData<List<String>> = _newsList
    init {
        // 模拟从网络数据库获取新闻数据
        fetchNews()
    }
    private fun fetchNews() {
        // 这里可以替换为真实的网络请求或数据库查询
        val mockNews = listOf("新闻1", "新闻2", "新闻3")
        _newsList.value = mockNews
    }
}
// 在 Activity 中使用 ViewModel
import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import androidx.lifecycle.ViewModelProvider
import kotlinx.android.synthetic.main.activity_main.*
class MainActivity : AppCompatActivity() {
    private lateinit var newsViewModel: NewsViewModel
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView{"name":"GodelPlugin","parameters":{"input":"\"setContentView(R.layout.activity_main)\""}}<|FunctionExecuteEnd|><|FunctionExecuteResult|>setContentView(R.layout.activity_main)<|FunctionExecuteResultEnd|>
        // 获取 ViewModel 实例
        newsViephpwModel = ViewModelProvider(this).get(NewsViewModel::class.Java)
        // 观察 LiveData 数据变化
        newsViewModel.newsList.observe(this, { news ->
            // 更新 UI
            news.forEach {
                textView.append("$it\n")
            }
        })
    }
}

当屏幕旋转等配置更改时,MainActivity 重新创建,但 NewsViewModel 实例会从 ViewModelStore 中取出,数据得以保留。 

LiveData 工作原理

  • 数据持有与观察者管理LiveData 内部维护着一个数据对象和一个观察者列表。当调用 observe 方法注册观察者时,会将 LifecycleOwner 和 Observer 包装成 LifecycleBoundObserver 对象并添加到观察者列表中。
  • 生命周期感知LifecycleBoundObserver 实现了 LifecycleEventObserver 接口,能够监听 LifecycleOwner 的生命周期变化。当 LifecycleOwner 进入活跃状态(STARTED 或 RESUMED)时,LiveData 会将最新数据发送给该观察者;当 LifecycleOwner 进入销毁状态(DESTROYED)时,LiveData 会自动移除该观察者,避免内存泄漏。
  • 数据更新通知:当调用 setValue(主线程)或 postValue(子线程)方法更新数据时,LiveData 会检查所有观察者的生命周期状态,只有处于活跃状态的观察者才会收到 onChanged 方法的调用,从而更新 UI。

 LiveData 持有数据,通过 LifecycleBoundObserver 感知 LifecycleOwner 生命周期,仅在活跃状态时通知观察者。

import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import androidx.lifecycle.MutableLiveData
import androidx.lifecycle.Observer
import kotlinx.android.synthetic.main.activity_main.*
class MainActivity : AppCompatActivity() {
    private val liveData = MutableLiveData<String>()
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView{"name":"GodelPlugin","parameters":{"input":"\"setContentView(R.layout.activity_main)\""}}<|FunctionExecuteEnd|><|FunctionExecuteResult|>setContentView(R.layout.activity_main)<|FunctionExecuteResu编程China编程ltEnd|>
        // 注册观察者
        liveData.observe(this, Observer { data ->
            // 处理数据变化
            textView.text = data
        })
        // 更新数据
        liveData.value = "新数据"
    }
}

liveData.observe 注册时将 this(即 MainActivity 作为 LifecycleOwner)和 Observer 包装,当 MainActivity 处于活跃状态且 liveData 数据更新时,Observer 的 onChanged 方法被调用。 

Room 工作原理

  • 抽象层封装:Room 提供了一个抽象层,开发者通过定义实体类(使用 @Entity 注解)、数据访问对象(DAO,使用 @Dao 注解)和数据库类(使用 @Database 注解)来描述数据库结构和操作。实体类对应数据库表,DAO 定义了对数据库的增删改查操作,数据库类则管理数据库的版本和 DAO 实例。
  • 编译时处理:在编译时,Room 会根据开发者定义的注解生成相应的 SQLite 语句和实现代码。这样可以在编译阶段就发现数据库操作中的错误,提高开发效率和代码的健壮性。
  • 线程管理:Room 默认不允许在主线程中执行数据库操作,因为数据库操作通常是耗时的,可能会导致 UI 卡顿。因此,Room 会将数据库操作放在后台线程中执行,开发者可以使用 suspend 函数(在 Kotlin 中)或自定义线程池来处理异步操作。

 Room 通过注解定义数据库结构和操作,编译时生成 SQL 语句和实现代码,默认在后台线程执行操作。

// 定义实体类
import androidx.room.Entity
import androidx.room.PrimaryKey
@Entity(tableName = "news")
data class News(
    @PrimaryKey(autoGenerate = true)
    val id: Int = 0,
    val title: String
)
// 定义 DAO
import androidx.room.Dao
import androidx.room.Insert
import androidx.room.Query
@Dao
interface NewsDao {
    @Insert
    suspend fun insertNews(news: News)
    @Query("SELECT * FROM news")
    suspend fun getAllNews(): List<News>
}
// 定义数据库类
import androidx.room.Database
import andChina编程roidx.room.RoomDatabase
@Database(entities = [News::class], version = 1)
abstract class AppDatabase : RoomDatabase() {
    abstract fun newsDao(): NewsDao
}
// 在 ViewModel 中使用 Room
import androidx.lifecycle.ViewModel
import androidx.lifecycle.viewModelScope
import kotlinx.coroutines.launch
class NewsViewModel : ViewModel() {
    private val database = Room.databaseBuilder(
        applicationContext,
        AppDatabase::class.java,
        "news-database"
    ).build()
    private val newsDao = database.newsDao()
    fun insertNews(news: News) {
        viewModelScope.launch {
            newsDao.insertNews(news)
        }
    }
    fun getAllNews() {
        viewModelScope.launch {
            val newsList = newsDao.getAllNews()
            // 处理获取到的新闻列表
        }
    }
}

编译时,Room 会根据 @Entity@Dao 和 @Database 注解生成操作数据库的 SQL 语句和实现代码,suspend 函数保证数据库操作在后台线程执行。 

Navigation 工作原理

  • 导航图定义:开发者通过 XML 文件定义导航图,导航图中包含了应用的所有目的地(如 Fragment)、动作(用于在目的地之间导航)和参数传递规则。每个目的地都有唯一的标识符,动作则定义了从一个目的地到另一个目的地的导航路径。
  • 导航控制器管理NavController 是 Navigation 组件的核心,负责管理导航操作。它会根据导航图中的定义,处理目的地之间的切换和参数传递。在 Activity 或 Fragment 中,可以通过 findNavController 方法获取 NavController 实例,然后调用其 navigate 方法进行导航。
  • Back Stack 管理NavController 维护了一个返回栈(Back Stack),用于记录导航历史。当用户点击返回按钮时,NavController 会从返回栈中弹出上一个目的地,实现返回操作。开发者可以通过配置导航图中的 popUpTo 和 popUpToInclusive 属性来控制返回栈的行为。

Navigation 通过导航图定义目的地和动作,NavController 管理导航和返回栈。

<navigation xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/navigation_graph"
    app:startDestination="@id/firstFragment">
    <fragment
   javascript     android:id="@+id/firstFragment"
        android:name="com.example.myapp.FirstFragment"
        android:label="First Fragment">
        <action
            android:id="@+id/action_firstFragment_to_secondFragment"
            app:destination="@id/secondFragment" />
    </fragment>
    <fragment
        android:id="@+id/secondFragment"
        android:name="com.example.myapp.SecondFragment"
        android:label="Second Fragment" />
</navigation>

在 Activity 中设置导航宿主

<androidx.fragment.app.FragmentContainerView
    android:id="@+id/nav_host_fragment"
    android:name="androidx.navigation.fragment.NavHostFragment"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    app:defaultNavHost="true"
    app:navGraph="@navigation/navigation_graph" />

在 Fragment 中进行导航

import androidx.fragment.app.Fragment
import android.os.Bundle
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import androidx.navigation.fragment.findNavController
class FirstFragment : Fragment() {
    override fun onCreateView(
        inflater: LayoutInflater, container: ViewGroup?,
        savedInstanceState: Bundle?
    ): View? {
        val view = inflater.inflate(R.layout.fragment_first, container, false)
        view.findViewById<Button>(R.id.navigateButton).setOnClickListener {
            // 导航到 SecondFragment
            findNavController().navigate(R.id.action_firstFragment_to_secondFragment)
        }
        return view
    }
}

 NavController 根据导航图中的定义,处理从 FirstFragment 到 SecondFragment 的导航,同时管理返回栈以支持返回操作。

 总结:

  ViewModel 通过 ViewModelStore 管理 UI 数据并在配置变更时保持状态,LiveData 实现生命周期感知的可观察数据更新,Room 作为 SQLite ORM 自动生成数据库操作代码并处理线程,Navigation 利用导航图和 NavController 管理多 Fragment 导航,共同构建响应式、可维护的 Android 应用架构。

到此这篇关于kotlin中的模块化结构组件的文章就介绍到这了,更多相关kotlin模块化结构组件内容请搜索China编程(www.chinasem.cn)以前的文章或继续浏览下面的相关文章希望大家以后多多支持China编程(www.chinasem.cn)!

这篇关于kotlin中的模块化结构组件及工作原理的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1153762

相关文章

Go 语言中的select语句详解及工作原理

《Go语言中的select语句详解及工作原理》在Go语言中,select语句是用于处理多个通道(channel)操作的一种控制结构,它类似于switch语句,本文给大家介绍Go语言中的select语... 目录Go 语言中的 select 是做什么的基本功能语法工作原理示例示例 1:监听多个通道示例 2:带

鸿蒙中@State的原理使用详解(HarmonyOS 5)

《鸿蒙中@State的原理使用详解(HarmonyOS5)》@State是HarmonyOSArkTS框架中用于管理组件状态的核心装饰器,其核心作用是实现数据驱动UI的响应式编程模式,本文给大家介绍... 目录一、@State在鸿蒙中是做什么的?二、@Spythontate的基本原理1. 依赖关系的收集2.

Python通过模块化开发优化代码的技巧分享

《Python通过模块化开发优化代码的技巧分享》模块化开发就是把代码拆成一个个“零件”,该封装封装,该拆分拆分,下面小编就来和大家简单聊聊python如何用模块化开发进行代码优化吧... 目录什么是模块化开发如何拆分代码改进版:拆分成模块让模块更强大:使用 __init__.py你一定会遇到的问题模www.

kotlin中const 和val的区别及使用场景分析

《kotlin中const和val的区别及使用场景分析》在Kotlin中,const和val都是用来声明常量的,但它们的使用场景和功能有所不同,下面给大家介绍kotlin中const和val的区别,... 目录kotlin中const 和val的区别1. val:2. const:二 代码示例1 Java

Java编译生成多个.class文件的原理和作用

《Java编译生成多个.class文件的原理和作用》作为一名经验丰富的开发者,在Java项目中执行编译后,可能会发现一个.java源文件有时会产生多个.class文件,从技术实现层面详细剖析这一现象... 目录一、内部类机制与.class文件生成成员内部类(常规内部类)局部内部类(方法内部类)匿名内部类二、

Kotlin 作用域函数apply、let、run、with、also使用指南

《Kotlin作用域函数apply、let、run、with、also使用指南》在Kotlin开发中,作用域函数(ScopeFunctions)是一组能让代码更简洁、更函数式的高阶函数,本文将... 目录一、引言:为什么需要作用域函数?二、作用域函China编程数详解1. apply:对象配置的 “流式构建器”最

Python中随机休眠技术原理与应用详解

《Python中随机休眠技术原理与应用详解》在编程中,让程序暂停执行特定时间是常见需求,当需要引入不确定性时,随机休眠就成为关键技巧,下面我们就来看看Python中随机休眠技术的具体实现与应用吧... 目录引言一、实现原理与基础方法1.1 核心函数解析1.2 基础实现模板1.3 整数版实现二、典型应用场景2

Java的IO模型、Netty原理解析

《Java的IO模型、Netty原理解析》Java的I/O是以流的方式进行数据输入输出的,Java的类库涉及很多领域的IO内容:标准的输入输出,文件的操作、网络上的数据传输流、字符串流、对象流等,这篇... 目录1.什么是IO2.同步与异步、阻塞与非阻塞3.三种IO模型BIO(blocking I/O)NI

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

kotlin的函数forEach示例详解

《kotlin的函数forEach示例详解》在Kotlin中,forEach是一个高阶函数,用于遍历集合中的每个元素并对其执行指定的操作,它的核心特点是简洁、函数式,适用于需要遍历集合且无需返回值的场... 目录一、基本用法1️⃣ 遍历集合2️⃣ 遍历数组3️⃣ 遍历 Map二、与 for 循环的区别三、高