Android-&-Kotlin:Retrofit-+-Hilt-手写实现—看妹子app(猛男必备)

//Room
def room_version = “2.2.5”
implementation “androidx.room:room-runtime: r o o m v e r s i o n " i m p l e m e n t a t i o n " a n d r o i d x . r o o m : r o o m − k t x : room_version" implementation "androidx.room:room-ktx: roomversion"implementation"androidx.room:roomktx:room_version”
kapt “androidx.room:room-compiler:$room_version”

//Navigation
def nav_version = “2.3.0”
implementation “androidx.navigation:navigation-fragment-ktx: n a v v e r s i o n " i m p l e m e n t a t i o n " a n d r o i d x . n a v i g a t i o n : n a v i g a t i o n − u i − k t x : nav_version" implementation "androidx.navigation:navigation-ui-ktx: navversion"implementation"androidx.navigation:navigationuiktx:nav_version”

//Glide 图片处理
implementation ‘com.github.bumptech.glide:glide:4.11.0’
kapt ‘com.github.bumptech.glide:compiler:4.11.0’

//Timber
implementation ‘com.jakewharton.timber:timber:4.7.1’
}

2 主要实现

  • 通过Navigation实现Fragment 之间数据传递
  • Hilt 实现依赖注入
  • 访问数据实现网络本地访问策略

3. 数据服务

分为两部分:

  • 网络数据remote
  • 本地数据local

没有数据的话网络请求,并将数据放入android的sqlite数据库,再次请求的时候查看本地数据库中是否存在数据如果存在的话就直接在本地获取,如果不存在从网络获取。

3.1 数据来源

数据来源: 干活集中营的api,api可以获取一些技术干货,也有一些福利妹子图片,本篇主要使用获取妹子图片这一部分

  • 看一下api内容:

  • 通过api内容写entity文件,获取自己需要的字段

/**

  • @author: ffzs

  • @Date: 2020/9/12 下午3:23
    */
    @Entity(tableName = “image_data_table”)
    data class Image (
    @PrimaryKey
    val _id:String,
    val author:String,
    var url:String,
    val title:String,
    val desc:String,
    val likeCounts:Long,
    val views:Long,
    )

  • 同时创建一个接收类,我们只需要获取response中的data:

/**

  • @author: ffzs
  • @Date: 20-9-12 下午8:15
    */

data class ImageList(
val data:List
)

3.2 网络获取数据

  • 配置Retrofit

  • api接口获取20个小姐姐和一个小姐姐

  • 获取数据并进行封装为Resource逻辑,Response中数据->Resource
  • 由于获取的图片网址为http协议,需要换成https不然无法跳转

protected suspend fun getResult(call: suspend () -> Response): Resource {
try {
val response = call()
if (response.isSuccessful) {
val body = response.body()
body as ImageList
// 将api获取的图片信息中http换为https不然无法完成跳转
body.data.map {
it.url = it.url.replace(“http://”, “https://”)
}
Timber.i(body.toString())
return Resource.success(body)
}
return error(" ${response.code()} ${response.message()}")
} catch (e: Exception) {
return error(e.message ?: e.toString())
}
}

3.3 本地数据获取

本地使用Room完成对sqlite的操作,主要有功能实现如下:

  • 网络获取数据缓存到本地, 需要进行插入处理
  • 获取数据优先本地缓存中获取,并通过views进行排序

/**

  • @author: ffzs
  • @Date: 2020/9/12 下午3:28
    */
    @Dao
    interface ImageDao {

@Query(“SELECT * FROM image_data_table ORDER BY views DESC”)
fun getAllImages() : LiveData<List>

@Query(“SELECT * FROM image_data_table WHERE _id = :id”)
fun getImage(id: String): LiveData

@Insert(onConflict = OnConflictStrategy.REPLACE)
suspend fun insertAll(image_data_table: List)

@Insert(onConflict = OnConflictStrategy.REPLACE)
suspend fun insert(image: Image)
}

  • Room Database配置
  • 要加上fallbackToDestructiveMigration()不处理Migration的话升级版本会报错

/**

  • @author: ffzs
  • @Date: 2020/9/12 下午3:27
    */

@Database(entities = [Image::class], version = 4)
abstract class ImageDatabase :RoomDatabase(){
abstract val imageDao: ImageDao

companion object {
@Volatile private var instance: ImageDatabase? = null

fun getDatabase(context: Context): ImageDatabase =
instance ?: synchronized(this) { instance ?: getInstance(context).also { instance = it } }

private fun getInstance(appContext: Context) =
Room.databaseBuilder(appContext, ImageDatabase::class.java, “image_data_table”)
.fallbackToDestructiveMigration()
.build()
}
}

3.4 获取数据

这里编写一个策略处理网络获取和本地获取的逻辑:

  • 本地获取成功,使用本地的,不成功网络获取并存储到本地
  • 这里传入的是三个方法

fun <T, A> performGetOperation(databaseQuery: () -> LiveData,
networkCall: suspend () -> Resource,
saveCallResult: suspend (A) -> Unit): LiveData<Resource> =
liveData(Dispatchers.IO) {
emit(Resource.loading())
val source = databaseQuery.invoke().map { Resource.success(it) }
emitSource(source)

val responseStatus = networkCall.invoke()
if (responseStatus.status == SUCCESS) {
saveCallResult(responseStatus.data!!)

} else if (responseStatus.status == ERROR) {
emit(Resource.error(responseStatus.message!!))
emitSource(source)
}
}

  • 通过Repository统一编写处理逻辑

fun getImages() = performGetOperation(
databaseQuery = { localDataSource.getAllImages() },
networkCall = { webDataSource.getImages() },
saveCallResult = { localDataSource.insertAll(it.data) }
)

4. Hilt 依赖注入

hilt官网

android开发文档

两部分:

  • Hilt 启动类
  • Hilt 组件

启用插件需要将classpath添加到依赖中

4.1 Hilt 启动类

我们首先在根文件夹中创建一个从Application继承的类,以对其进行注释,以告知我们将在应用程序中使用Hilt。该类为Application类

当有了Application, 我们可以讲其他Android类中的启动成员注入,使用@AndroidEntryPoints注释。

@AndroidEntryPoint在以下类型上使用:

  1. Activity
  2. Fragment
  3. View
  4. Service
  5. BroadcastReceiver

4.2 Hilt 组件注入

使用 @Provides 注入

您可以告知 Hilt 如何提供此类型的实例,方法是在 Hilt 模块内创建一个函数,并使用 @Provides 为该函数添加注释。

带有注释的函数会向 Hilt 提供以下信息:

  • 函数返回类型会告知 Hilt 函数提供哪个类型的实例。
  • 函数参数会告知 Hilt 相应类型的依赖项。
  • 函数主体会告知 Hilt 如何提供相应类型的实例。每当需要提供该类型的实例时,Hilt 都会执行函数主体。

5. Fragment操作

通过Fragment以及navigation完成界面的切换

5.1 src/main/res/navigation/nav_graph.xml:

  • main中通过FragmentContainerView绑定到navigation
  • navigation实现两个界面的绑定
  • 通过一个action进行切换
  • 切换的同时将list中的id进行传递,detail中获取id后访问数据库获取数据进行展示
<?xml version="1.0" encoding="utf-8"?>

<navigation xmlns:android=“http://schemas.android.com/apk/res/android”
xmlns:app=“http://schemas.android.com/apk/res-auto”
xmlns:tools=“http://schemas.android.com/tools”
android:id=“@+id/nav_graph”
app:startDestination=“@id/imageFragment”

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

w-1714961287567)]

[外链图片转存中…(img-yQ1YWLnc-1714961287568)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

  • 17
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值