Android kotlin Jetpack mvvm 项目

<?xml version="1.0" encoding="utf-8"?>





  • 还可以切换到Design模式编辑

在这里插入图片描述 在这里插入图片描述

NavController
  • 在fragment1中直接调用以下代码即可从fragment1切换到fragment2,并且传入参数

findNavController().navigate(NavigationFragment1Directions.actionNavigationFragment1ToNavigationFragment22(“测试”))

其他功能
  • 全局操作 使用全局操作来创建可由多个目的地共用的通用操作
  • 创建深层链接 将用户直接转到应用内特定目的地的链接
  • 使用 NavigationUI 更新界面组件 使用顶部应用栏、抽屉式导航栏和底部导航栏来管理导航
  • 自定义返回导航

Lifecycle

传统在生命周期方法写逻辑代码的缺点
  • 太多管理界面和其他组件的调用,以响应生命周期的当前状态。管理多个组件会在生命周期方法(如 onStart() 和 onStop())中放置大量的代码,这使得它们难以维护
  • 无法保证组件会在 Activity 或 Fragment 停止之前启动。在我们需要执行长时间运行的操作(如 onStart() 中的某种配置检查)时尤其如此
Lifecycle的优势
  • 生命周期感知型组件可执行操作来响应另一个组件(如 Activity 和 Fragment)的生命周期状态的变化。这些组件有助于您编写出更有条理且往往更精简的代码,您可以将依赖组件的代码从生命周期方法移入组件本身中,此类代码更易于维护。
lifecycle使用
  • 定义LifecycleObserver

inner class MyLifecycleObserver(val lifecycle: Lifecycle) : LifecycleObserver {

@OnLifecycleEvent(Lifecycle.Event.ON_START)
fun onStart() {
CoroutineScope(scope).launch {
delay(3000)
if (lifecycle.currentState.isAtLeast(Lifecycle.State.STARTED)) {
LogUtil.e(“开启定位”)
}
}
}

@OnLifecycleEvent(Lifecycle.Event.ON_STOP)
fun onStop() {
LogUtil.e(“关闭定位”)
}
}

  • 添加LifecycleObserver

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
lifecycle.addObserver(MyLifecycleObserver(lifecycle))
}

ViewModel

  • 旨在以注重生命周期的方式存储和管理界面相关的数据。ViewModel 类让数据可在发生屏幕旋转等配置更改后继续留存,负责为界面准备数据。在配置更改期间会自动保留 ViewModel 对象,以便它们存储的数据立即可供下一个 Activity 或 Fragment 实例使用。
ViewModel可以解决什么问题
  • 如果系统销毁或重新创建界面控制器,则存储在其中的任何瞬态界面相关数据都会丢失,为了避免我们的程序出现上述这种情况,我们除了使用Activity的savedInstanceState(仅适合可以序列化再反序列化的少量数据)保存数据之外,还可以使用ViewModel来进行处理数据(可保存较大数据)
  • 简化资源管理工作,避免内存泄漏风险,Activity 和 Fragment经常需要进行可能需要一些时间才能返回的异步调用(如网络请求),我们需要确保系统在其销毁后清理这些调用以避免潜在的内存泄漏,同时如果配置发生更改重新创建对象的时候可能会发生重复此前已完成的工作,造成资源浪费
  • 让 Activity 和 Fragment 专注于界面显示,如果要求界面控制器也负责从数据库或网络加载数据,那么会使类越发膨胀,ViewModel可以更容易、高效的分离出视图数据操作
  • 实现Fragment和Activity/Fragment共享数据
ViewModel使用
  • 实现ViewModel,继承ViewModel抽象类即可

class ViewModelViewModel: ViewModel() {
val userList = mutableListOf()
}

  • 引用ViewModel

在fragment/activity中使用

val viewModel:ViewModelViewModel by viewModels()
viewModel.userList

在多个fragment中共享的ViewModel,使用by activityViewModels()创建的ViewModel依赖于Activity,在多个Fragment为同一个对象

val shareViewModel:ViewModelViewModel by activityViewModels()

或者使用koin依赖注入,koin使用说明在下文

val viewModel: ViewModelViewModel by viewModel()

val viewModel: ViewModelViewModel by sharedViewModel()

ViewModel生命周期

在这里插入图片描述

生命周期感知数据 LiveData

是一种可观察的数据存储器类。与常规的可观察类不同,LiveData 具有生命周期感知能力,意指它遵循其他应用组件(如 Activity、Fragment 或 Service)的生命周期。这种感知能力可确保 LiveData 仅更新处于活跃生命周期状态的应用组件观察者。

LiveData的优势
  • 确保界面符合数据状态:LiveData 遵循观察者模式。当底层数据发生变化时,LiveData 会通知 Observer 对象。您可以整合代码以在这些 Observer 对象中更新界面。
  • 不会发生内存泄漏:观察者会绑定到 Lifecycle 对象,并在其关联的生命周期遭到销毁后进行自我清理。
  • 不会因 Activity 停止而导致崩溃:如果观察者的生命周期处于非活跃状态(如返回栈中的 Activity),则它不会接收任何 LiveData 事件。
  • 不再需要手动处理生命周期:界面组件只是观察相关数据,不会停止或恢复观察。LiveData 将自动管理所有这些操作,因为它在观察时可以感知相关的生命周期状态变化。
  • 数据始终保持最新状态:如果生命周期变为非活跃状态,它会在再次变为活跃状态时接收最新的数据。
  • 适当的配置更改:如果由于配置更改(如设备旋转)而重新创建了 Activity 或 Fragment,它会立即接收最新的可用数据。
  • 共享资源:可以使用单例模式扩展 LiveData 对象以封装系统服务,以便在应用中共享它们。
LiveData使用
  • 定义

private val normaLiveData1 = MutableLiveData()

  • 赋值

normaLiveData1.value=“LiveDataValue”//UI线程
normaLiveData1.postValue(“LiveDataValue”)//非UI主线程

  • 观察数据

normaLiveData1.observe(viewLifecycleOwner, Observer {
LogUtil.e(“观察到第一个值发生了变化”)
tv.text = it
})

  • 无生命周期感知地观察数据,这种方式需要手动取消观察,否则会发生内存泄漏

val observer=Observer{
LogUtil.e(“观察到第一个值发生了变化”)
}
normaLiveData1.observeForever(observer)
//在合适的生命周期移除观察
normaLiveData1.removeObserver(observer)

  • 转换 LiveData

private val transLiveData= Transformations.map(normaLiveData1){
“$it -----转换”
}

  • 合并多个 LiveData

private val mediatorLiveData = MediatorLiveData()

mediatorLiveData.addSource(normaLiveData1){
mediatorLiveData.value=“合并后的值: i t − − − it--- it{normaLiveData2.value}”
}

mediatorLiveData.addSource(normaLiveData2){
mediatorLiveData.value=“合并后的值: n o r m a L i v e D a t a 1. v a l u e − − − {normaLiveData1.value}--- normaLiveData1.valueit”
}

  • LiveData结合room、协程使用(下文)

kotlin数据流 Flow

Flow数据流以协程为基础构建,可提供多个值。从概念上来讲,数据流是可通过异步方式进行计算处理的一组数据序列,有点像RxJava。

Flow的使用
  • 创建Flow

val flow = flow {
emit(“value1”)
emit(“value2”)
delay(1000)
emit(“value3”)
}

val flow = flowOf(“value”)

val flow = listOf(1, 2, 3).asFlow()

  • 收集Flow----collect()

由于 collect 是挂起函数,因此需要在协程中执行

scope.launch {
flow.collect {
LogUtil.e(it)
}
}

  • 转换Flow----map()

flowOf(1, 2, 3).map {
“第$it 个”
}.collect {
LogUtil.e(it)
}

  • 过滤Flow----filter()

flowOf(1, 2, 3).filter {
it > 1
}.collect {
LogUtil.e(it)
}

  • 合并Flow

zip操作符会把 flow1 中的一个 item 和 flow2 中对应的一个 item 进行合并,如果 flow1 中 item 个数大于 flow2 中 item 个数,合并后新的 flow 的 item 个数 = 较小的 flow 的 item 个数

val flow1 = flowOf(1, 2, 3, 4, 5)
val flow2 = flowOf(“一”, “二”, “三”, “四”, “五”, “六”)
flow1.zip(flow2) { a, b ->
a − − − a--- ab”
}.collect {
LogUtil.e(it)
}

combine合并时,每次从 flow1 发出新的 item ,会将其与 flow2 的最新的 item 合并

val flow1 = flowOf(1, 2, 3, 4, 5).onEach { delay(1000) }
val flow2 = flowOf(“一”, “二”, “三”, “四”, “五”, “六”).onEach { delay(500) }
flow1.combine(flow2) { a, b ->
a − − − a--- ab”
}.collect {
LogUtil.e(it)
}

  • 捕获异常----catch()

flow {
emit(1)
emit(1 / 0)
emit(2)
}.catch {
it.printStackTrace()
}.collect {
LogUtil.e(it)
}

  • 线程切换----flowOn()

withContext(Dispatchers.IO){
flowOf(1, 2, 3, 4).onEach {
//受到下面最近的flowOn控制-Main
LogUtil.e(“init—KaTeX parse error: Expected 'EOF', got '}' at position 33: …read().name}") }̲.filter { //受到下…{Thread.currentThread().name}”)
it > 1
}.flowOn(Dispatchers.Main).map {
//受到下面最近的flowOn控制-IO
LogUtil.e(“map— T h r e a d . c u r r e n t T h r e a d ( ) . n a m e " ) " 第 {Thread.currentThread().name}") "第 Thread.currentThread().name")"it”
}.flowOn(Dispatchers.IO).map {
//受到下面最近的flowOn控制-Main
LogUtil.e(“第二次map— T h r e a d . c u r r e n t T h r e a d ( ) . n a m e " ) " {Thread.currentThread().name}") " Thread.currentThread().name")"it 个结果”
}.flowOn(Dispatchers.Main).collect {
//collect要看整个flow处于哪个线程,此处为IO
LogUtil.e(“collect—${Thread.currentThread().name}”)
LogUtil.e(it)
}
}

  • 转为liveData----asLiveData()

添加依赖

“androidx.lifecycle:lifecycle-livedata-ktx:${LibraryVersion.LIVEDATA_KTX}”

flowOf(1, 2, 3, 4).asLiveData().observe(viewLifecycleOwner, Observer {
LogUtil.e(it)
})

更多操作符

本地保存 DataStore

DataStore 是一种数据存储解决方案,允许您使用协议缓冲区存储键值对或类型化对象。DataStore 使用 Kotlin 协程和 Flow 以异步、一致的事务方式存储数据。 参考鸿洋的公众号内容

SharedPreferences存在的问题
  • 通过 getXXX() 方法获取数据,可能会导致主线程阻塞
  • SharedPreference 不能保证类型安全
  • SharedPreference 加载的数据会一直留在内存中,浪费内存
  • apply() 方法虽然是异步的,可能会发生 ANR,在 8.0 之前和 8.0 之后实现各不相同
  • apply() 方法无法获取到操作成功或者失败的结果
DataStore 解决了什么问题
  • DataStore 是基于 Flow 实现的,所以保证了在主线程的安全性
  • 以事务方式处理更新数据,事务有四大特性(原子性、一致性、 隔离性、持久性)
  • 没有 apply() 和 commit() 等等数据持久的方法
  • 自动完成 SharedPreferences 迁移到 DataStore,保证数据一致性,不会造成数据损坏
  • 可以监听到操作成功或者失败结果
  • 另外 Jetpack DataStore 提供了 Proto DataStore 方式,用于存储类的对象(typed objects ),通过 protocol buffers 将对象序列化存储在本地,protocol buffers 现在已经应用的非常广泛,无论是微信还是阿里等等大厂都在使用
DataStore使用

添加依赖

const val DATA_STORE = “1.0.0-alpha05”
const val PROTOBUF = “3.11.0”
“androidx.datastore:datastore-preferences: L i b r a r y V e r s i o n . D A T A S T O R E " / / p r o t o b u f 需下面的依赖 " a n d r o i d x . d a t a s t o r e : : d a t a s t o r e − c o r e : {LibraryVersion.DATA_STORE}" //protobuf需下面的依赖 "androidx.datastore::datastore-core: LibraryVersion.DATASTORE"//protobuf需下面的依赖"androidx.datastore::datastorecore:{LibraryVersion.DATA_STORE}”
“com.google.protobuf:protobuf-java:${LibraryVersion.PROTOBUF}”

保存键值对

object DataStore {

private const val APP_DATA_STORE_NAME = “APP_DATA_STORE_NAME”
private lateinit var dataStore: DataStore

fun init(context: Context) {
dataStore = context.createDataStore(APP_DATA_STORE_NAME)
}

suspend fun save(key: Preferences.Key, value: T) {
dataStore.edit {
it[key] = value
}
}

suspend fun get(key: Preferences.Key): T? {
val value = dataStore.data.map {
it[key]
}
return value.first()
}

}

保存

CoroutineScope(scope).launch {
DataStore.save(preferencesKey(“key1”), “aa”)
}

读取

CoroutineScope(scope).launch {
val get = DataStore.get(preferencesKey(“key1”))
}

保存protobuf

protobuf相关知识不在这里展开叙述

  • 定义.proto文件

syntax = “proto3”;

option java_package = “com.haikun.jetpackapp.home.ui.demo.datastore.bean”;
option java_multiple_files = true;

message MessageEvent {
int32 type = 1;
string message = 2;
}

  • 编译文件

编译后得到下图三个文件

在这里插入图片描述

  • 定义Serializer

object MessageSerializer : Serializer {
override val defaultValue: MessageEvent
get() = MessageEvent.getDefaultInstance()

override fun readFrom(input: InputStream): MessageEvent {
return MessageEvent.parseFrom(input)
}

override fun writeTo(t: MessageEvent, output: OutputStream) {
t.writeTo(output)
}
}

  • 保存

val createDataStore = context?.createDataStore(“data”, MessageSerializer)
createDataStore?.updateData {
it.toBuilder().setType(12).setMessage(“消息”).build()
}

  • 读取

CoroutineScope(scope).launch {
context?.createDataStore(“data”, MessageSerializer)?.data?.first()?.let {
LogUtil.e(“ i t . t y p e − − − {it.type}--- it.type{it.message}”)
}
}

声明式UI DataBinding

数据绑定库是一种支持库,借助该库,您可以使用声明性格式(而非程序化地)将布局中的界面组件绑定到应用中的数据源,可以使用 LiveData 对象作为数据绑定来源,自动将数据变化通知给界面

声明式UI VS 命令式UI
  • 声明式UI 只需要把界面给「声明」出来,而不需要手动更新,只要声明的数据发生了变化,UI就跟着变化

  • 命令式UI 需要主动让UI更新,比如setText()

DataBinding使用
开启DataBinding

android {

dataBinding {
enabled = true
}
}

基本用法
  • 在ViewModel中定义数据和方法

class DataBindingViewModel : ViewModel() {

val userName = MutableLiveData()
val clickTimes = MutableLiveData()
val sexCheckId = MutableLiveData()
val love = MutableLiveData()

fun save(){
LogUtil.e(“ u s e r N a m e . v a l u e − − − {userName.value}--- userName.value{sex.value}—${love.value}”)
}
}

  • xml中引用和调用



  • Fragment

class DataBindingFragment : Fragment() {

private val mViewModel: DataBindingViewModel by viewModels()

override fun onCreateView(
inflater: LayoutInflater,
container: ViewGroup?,
savedInstanceState: Bundle?
): View {
val dataBinding = DataBindingUtil.inflate(
inflater,
R.layout.fragment_data_binding,
container,
false
)
//使用liveData必须要设置lifecycleOwner,否则无法更新数据
dataBinding.lifecycleOwner = viewLifecycleOwner
dataBinding.viewModel = mViewModel
return dataBinding.root

}
}

进阶用法
  • BindingMethods 绑定方法名

@BindingMethods(value = [BindingMethod(type = MyButton::class, attribute = “maxTimes”, method = “setMaxTimes”)])

xml中使用

app:maxTimes=“@{15}”

  • BindingAdapter 提供自定义逻辑

一些属性需要自定义绑定逻辑。例如,android:paddingLeft 特性没有关联的 setter,而是提供了 setPadding(left, top, right, bottom) 方法。使用 BindingAdapter 注释的静态绑定适配器方法支持自定义特性 setter 的调用方式。

object ViewAdapter {
@BindingAdapter(“minTimes”)
@JvmStatic
fun setMinTimes(view: MyButton, minTimes: Int) {
view.setMin(minTimes)
}
}

xml中使用

app:minTimes=“@{8}”

  • 自定义双向绑定

@InverseBindingAdapter(attribute = “clickTimes”)
@JvmStatic
fun getClickTimes(view: MyButton): Int {
return view.clickTimes
}

@BindingAdapter(“clickTimesAttrChanged”)
@JvmStatic
fun setListener(view: MyButton, listener: InverseBindingListener?) {

view.onTimesChangeListener = {
listener?.onChange()
}

}

xml使用

app:clickTimes=“@={viewModel.clickTimes}”

Compose----android声明式UI未来的趋势
  • 2019 年中,Google 在 I/O 大会上公布了 Android 最新的 UI 框架:Jetpack Compose。Compose 可以说是 Android 官方有史以来动作最大的一个库了。它在 2019 年中就公布了,但要到今年也就是 2021 年才会正式发布。这两年的时间 Android 团队在干嘛?在开发这个库,在开发 Compose。一个 UI 框架而已,为什么要花两年来打造呢?因为 Compose 并不是像 RecyclerView、ConstraintLayout 这种做了一个或者几个高级的 UI 控件,而是直接抛弃了我们写了 N 年的 View 和 ViewGroup 那一套东西,从上到下撸了一整套全新的 UI 框架。直白点说就是,它的渲染机制、布局机制、触摸算法以及 UI 的具体写法,全都是新的。
  • 第一眼看到Compose,第一感觉就是觉得和Flutter的写法惊人的相似,Compose需要系统的学习,而且需要较高的学习成本,不在这里展开叙述
  • Compose只支持kotlin,并且目前需要用Canary版本的android studio进行开发

数据库 Room

Room 在 SQLite 上提供了一个抽象层,以便在充分利用 SQLite 的强大功能的同时,能够流畅地访问数据库 Room可以和kotlin协程/flow结合使用 Room可以和LiveData结合使用

Room 包含 3 个主要组件:

  • 数据库:包含数据库持有者,并作为应用已保留的持久关系型数据的底层连接的主要接入点。
  • Entity:表示数据库中的表。
  • DAO:包含用于访问数据库的方法。
添加依赖

implementation “androidx.room:room-runtime: r o o m v e r s i o n " k a p t " a n d r o i d x . r o o m : r o o m − c o m p i l e r : room_version" kapt "androidx.room:room-compiler: roomversion"kapt"androidx.room:roomcompiler:room_version”

implementation “androidx.room:room-ktx:$room_version”

基本用法
  • 定义Entity

@Entity
data class Car(
@PrimaryKey(autoGenerate = true) val id: Long,
var name: String,
val color: String,
)

@PrimaryKey 主键 类名就是表名,也可以在@Entity(table=)设置表名 可以使用@Ignore忽略某个字段

  • 定义Dao

@Dao
interface CarDao {

@Insert(onConflict = OnConflictStrategy.REPLACE)
fun insertCar(car: Car):Long

@Delete
fun deleteCar(car: Car)

@Update
fun updateCar(car: Car)

@Query(“SELECT * From Car”)
fun queryCarList(): MutableList

@Query(“Select * from Car where id=:id”)
fun queryCarById(id: Long): Car?
}

  • 定义Database

@Database(entities = [Car::class], version = 1,exportSchema = false)
abstract class DemoDatabase : RoomDatabase() {

abstract fun carDao(): CarDao

}

  • 创建Database和获取Dao

private val db: DemoDatabase by lazy {
Room.databaseBuilder(
JetpackApp.getContext(),
DemoDatabase::class.java, “demo-database”
).build()
}

private val carDao: CarDao by lazy {
db.carDao()
}

  • 增删改查

carDao.insertCar(car)
carDao.delete(car)
carDao.updateCar(car)
val car = carDao.queryCarById(mUpdateId)

Room并不支持在主线程访问数据库, 除非在Builder调用allowMainThreadQueries()方法, 因为它很可能将UI锁上较长一段时间. 但是, 异步查询–返回LiveData/Flowable实例的查询–则从此规则中免除, 因为它们在需要的时候会在后台线程异步地运行查询.

  • 使用Flow流进行响应式查询

只要表中的任何数据发生变化,返回的 Flow 对象就会再次触发查询并重新发出整个结果集。

使用 Flow 的响应式查询有一个重要限制:只要对表中的任何行进行更新(无论该行是否在结果集中),Flow 对象就会重新运行查询。通过将 distinctUntilChanged() 运算符应用于返回的 Flow 对象,可以确保仅在实际查询结果发生更改时通知界面:

@Query(“Select * From Car where id = :id”)
fun queryCarAsFlowById(id: Long): Flow

fun queryCarAsFlowByIdDistinctUntilChanged(id: Long): Flow<Car?> =
queryCarAsFlowById(id).distinctUntilChanged()

  • 使用 Kotlin 协程进行异步查询

将 suspend Kotlin 关键字添加到 DAO 方法中,以使用 Kotlin 协程功能使这些方法成为异步方法。这样可确保不会在主线程上执行这些方法。

  • 使用 LiveData 进行可观察查询

@Query(“Select * From Car where id = :id”)
fun queryCarAsLiveDataById(id: Long): LiveData<Car?>

对象之间的关系
嵌套关系
一对一
一对多
多对多

以一对多为例

  • 定义Entity和关系

@Entity
data class One(@PrimaryKey(autoGenerate = true) val id: Long, val name: String)

@Entity
data class More(@PrimaryKey(autoGenerate = true) val id: Long, val oneId: Long, val name: String)

data class OneAndMore(
@Embedded val one: One,
@Relation(
parentColumn = “id”,
entityColumn = “oneId”
) val moreList: MutableList
)

  • 定义Dao

添加 @Transaction 注释,以确保整个操作以原子方式执行。

@Transaction
open fun insertOneAndMore(){
val one = One(0, “OneName”)
val insertOneId = insertOne(one)
val more = More(0, insertOneId, “moreName1”)
val more1 = More(0, insertOneId, “moreName2”)
insertMore(more)
insertMore(more1)
}

@Transaction
@Query(“Select * from One”)
abstract fun queryOneAndMore():MutableList

使用类型转换器处理复杂数据

有时需要使用自定义数据类型,其中包含想要存储到单个数据库列中的值。TypeConverter可以在自定义类与 Room 可以保留的已知类型之间来回转换。

例如需要把一个包含List的对象保存到数据库

  • 定义Entity

@Entity
data class ComplexEntity(
@PrimaryKey val id: Long,
val list: MutableList
)

  • 定义Converter

class Converters {
@TypeConverter
fun fromJson(value: String): MutableList? {
val types =
Types.newParameterizedType(MutableList::class.java, OneAndMore::class.java)
return MoshiInstance.moshi.adapter<MutableList>(types).fromJson(value)
}

@TypeConverter
fun toJson(list: MutableList): String {
val types =
Types.newParameterizedType(MutableList::class.java, OneAndMore::class.java)
return MoshiInstance.moshi.adapter<MutableList>(types).toJson(list)
}
}

  • 添加Converter到Database

@TypeConverters(Converters::class)
abstract class DemoDatabase : RoomDatabase()

  • Dao

@Insert
abstract fun insertComplexEntity(complexEntity: ComplexEntity)

@Query(“Select * from ComplexEntity”)
abstract fun queryComplexEntity():MutableList

依赖注入 Koin

参考 依赖项注入 (DI) 是一种广泛用于编程的技术,Hilt、Dagger、Koin 等等都是依赖注入库,依赖注入是面向对象设计中最好的架构模式之一,使用依赖注入库有以下优点:

  • 依赖注入库会自动释放不再使用的对象,减少资源的过度使用。
  • 在指定范围内,可重用依赖项和创建的实例,提高代码的可重用性,减少了很多模板代码。
  • 代码变得更具可读性。
  • 易于构建对象。
  • 编写低耦合代码,更容易测试。
为什么选择Koin(而不是Jetpack Hilt组件)
  • 纯Kotlin
  • Koin上手简单,Hilt 使用起来要比 Koin 麻烦很多,其入门门槛高于 Koin,依赖注入部分的代码 Hilt 多于 Koin,在一个更大更复杂的项目中所需要的代码也更多,也越来越复杂
  • Koin编译时间短,在大型项目这个很重要
  • 项目结构:完成 Hilt 的依赖注入需要的文件往往多于 Koin
  • 代码行数:Hilt 生成的代码多于 Koin,随着项目越来越复杂,生成的代码量会越来越多。
  • Koin 中不需要使用注解,也不需要 kapt,不需要反射,使用了 kotlin 强大的语法糖(例如 Inline、Reified 等等)和函数式编程
  • Hilt在动态模块FDM中需要使用Dagger支持,Koin可以在FDM中使用
添加依赖

const val KOIN_SCOPE = “org.koin:koin-androidx-scope: C o r e V e r s i o n . K O I N " c o n s t v a l K O I N V I E W M O D E L = " o r g . k o i n : k o i n − a n d r o i d x − v i e w m o d e l : {CoreVersion.KOIN}" const val KOIN_VIEWMODEL = "org.koin:koin-androidx-viewmodel: CoreVersion.KOIN"constvalKOINVIEWMODEL="org.koin:koinandroidxviewmodel:{CoreVersion.KOIN}”
const val KOIN_FRAGMENT = “org.koin:koin-androidx-fragment:${CoreVersion.KOIN}”

Android高级架构师

由于篇幅问题,我呢也将自己当前所在技术领域的各项知识点、工具、框架等汇总成一份技术路线图,还有一些架构进阶视频、全套学习PDF文件、面试文档、源码笔记。

  • 330页PDF Android学习核心笔记(内含上面8大板块)

  • Android学习的系统对应视频

  • Android进阶的系统对应学习资料

  • Android BAT部分大厂面试题(有解析)

好了,以上便是今天的分享,希望为各位朋友后续的学习提供方便。觉得内容不错,也欢迎多多分享给身边的朋友哈。

越复杂

  • Koin编译时间短,在大型项目这个很重要
  • 项目结构:完成 Hilt 的依赖注入需要的文件往往多于 Koin
  • 代码行数:Hilt 生成的代码多于 Koin,随着项目越来越复杂,生成的代码量会越来越多。
  • Koin 中不需要使用注解,也不需要 kapt,不需要反射,使用了 kotlin 强大的语法糖(例如 Inline、Reified 等等)和函数式编程
  • Hilt在动态模块FDM中需要使用Dagger支持,Koin可以在FDM中使用
添加依赖

const val KOIN_SCOPE = “org.koin:koin-androidx-scope: C o r e V e r s i o n . K O I N " c o n s t v a l K O I N V I E W M O D E L = " o r g . k o i n : k o i n − a n d r o i d x − v i e w m o d e l : {CoreVersion.KOIN}" const val KOIN_VIEWMODEL = "org.koin:koin-androidx-viewmodel: CoreVersion.KOIN"constvalKOINVIEWMODEL="org.koin:koinandroidxviewmodel:{CoreVersion.KOIN}”
const val KOIN_FRAGMENT = “org.koin:koin-androidx-fragment:${CoreVersion.KOIN}”

Android高级架构师

由于篇幅问题,我呢也将自己当前所在技术领域的各项知识点、工具、框架等汇总成一份技术路线图,还有一些架构进阶视频、全套学习PDF文件、面试文档、源码笔记。

  • 330页PDF Android学习核心笔记(内含上面8大板块)

[外链图片转存中…(img-spuiBLZZ-1718809452391)]

[外链图片转存中…(img-qgfJVxdL-1718809452391)]

  • Android学习的系统对应视频

  • Android进阶的系统对应学习资料

[外链图片转存中…(img-cvNoneM6-1718809452391)]

  • Android BAT部分大厂面试题(有解析)

[外链图片转存中…(img-VgAPYhdo-1718809452392)]

好了,以上便是今天的分享,希望为各位朋友后续的学习提供方便。觉得内容不错,也欢迎多多分享给身边的朋友哈。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值