Android & Kotlin:Retrofit + Hilt 实现 看妹子app Android - 控件抖动效果 DevEco studio 一直加载gradle android如何获取调试版及发布版安全码SHA1 Android手机获取IP地址的两种方法 Android Activity 启动过程详解(上) 思维破局:挣钱其实没那么难,关键是方式要对 大数据赋能全产业链 为市民提供“一杯好奶” 运用大数据实现节能减排精准化 2020最新拼多多Java面试版,五面拿到28K的Offer 中国电信(安徽)大数据产业园落户南岗科技园 打造华东区域云计算领头羊 深度解读!阿里腾讯滴滴字节首选,新一代大数据引擎Flink厉害在哪?附学习礼包 网络编程5:socket服务端和客户端代码实现 微信小程序 蓝牙重连异常 errCode:10004,errMsg:notifyBLECharacteristicValueChange:fail setNot 蓝桥杯--第八届省赛试题-电子钟程序设计 【TCP回声服务器】一篇文章帮助你看懂TCP! 关于国标GB28181流媒体协议视频平台EasyGBS设备无法接入平台的问题排查 机器学习16 -- Lifelong Learning 终生学习 Vuex中发送mock请求返回数据 JS 正则表达式详解 学习笔记 花了30分钟,给女朋友们写了个最近抖音很火的3D立体动态相册 这 16 个 CSS 伪类,助你提升布局效率! Mysql系列第八讲 详解排序和分页(order by & limit)及存在的坑 SQL语句的一些应用语句 介绍三个开发技术小知识点 数据库系统概论(1) Mysql备份和恢复的类型,备份恢复的几种方法! --个人整理 MySql数据库基本操作(一) 前八月融合创新提升服务,大数据搭建信息网 长三角税收一体化按下“快进键” 一些建议:给当年刚做程序员的我 谁是物联网江湖明日之星?答案是新CIoT 企业信用有效“变现” 江苏银行物联网金融规模达185亿 2020年中国5G物联网行业:中国通信运营商5G投资达1.23万亿元-可下载 搭建 GFS 分布式文件系统-----实操 排障集锦:九九八十一难之第十五难!-------------史上最全rsync报错详解 docker系列1---docker隔离和限制技术 Linux系统安全及应用加固———最适合新手学,新手都能看懂!超详细的理论+超详细的实验!呕心沥血之作完成! LAMP架构集成ownCloud搭建私有云盘 Shell脚本正则表达式(一) Linux第一人称侵入进程的好地方 Windows DOS常用命令(持续更新完善) 大数据驱动新材料研发 变革从内而发,一站式大数据平台和多模数据库雄起 雕佛像、打棒球、开飞机、去北极,这个90后程序员有点不一样 12项特色功能构建施工现场“物联网” 400亿美元!英伟达收购Arm 软银套现离场物联网梦醒 卫星物联网再次引发热议,原因在这里 信用飞:航旅行业数字化前景可期,大数据预防砍头息乱象 大数据加持自动驾驶产业发展 上海智能网联汽车公共数据中心揭牌 恒久科技(002808.SZ):子公司闽保信息与鸿达兴业签署智能制造及大数据平台综合应用项目战略合作框架协议
您的位置:首页 >程序人生 >

Android & Kotlin:Retrofit + Hilt 实现 看妹子app

今天来学习一下android网络数据的访问,以及使用hilt dagger进行组件注入的实现。

0.效果展示

在这里插入图片描述

1. 依赖

使用到的技术栈:

Android studio 4.01fragmentRetrofitLifecycleKotlin CoroutinesHiltRoomNavigationGlideTimber

完整dependencies如下:

dependencies {implementation fileTree(dir: 'libs', include: ['*.jar'])implementation"org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"implementation 'androidx.appcompat:appcompat:1.2.0'implementation 'androidx.core:core-ktx:1.3.1'implementation 'androidx.constraintlayout:constraintlayout:2.0.1'testImplementation 'junit:junit:4.12'androidTestImplementation 'androidx.test:runner:1.3.0'androidTestImplementation 'androidx.test.espresso:espresso-core:3.3.0'// fragmentimplementation 'androidx.fragment:fragment-ktx:1.2.5'//Retrofitimplementation 'com.squareup.retrofit2:retrofit:2.9.0'implementation 'com.squareup.retrofit2:converter-gson:2.9.0'//Lifecycledef lifecycle_version = "2.2.0"implementation "androidx.lifecycle:lifecycle-viewmodel-ktx:$lifecycle_version"implementation "androidx.lifecycle:lifecycle-livedata-ktx:$lifecycle_version"implementation "androidx.lifecycle:lifecycle-common-java8:$lifecycle_version"implementation "androidx.lifecycle:lifecycle-extensions:$lifecycle_version"//noinspection LifecycleAnnotationProcessorWithJava8kapt "androidx.lifecycle:lifecycle-compiler:$lifecycle_version"//Kotlin Coroutinesdef coroutines_android_version = '1.3.9'implementation "org.jetbrains.kotlinx:kotlinx-coroutines-core:$coroutines_android_version"implementation "org.jetbrains.kotlinx:kotlinx-coroutines-android:$coroutines_android_version"//Hilt 依赖注入implementation 'com.google.dagger:hilt-android:2.28.1-alpha'implementation "androidx.hilt:hilt-lifecycle-viewmodel:1.0.0-alpha02"kapt 'com.google.dagger:hilt-android-compiler:2.28.1-alpha'kapt "androidx.hilt:hilt-compiler:1.0.0-alpha02"//Roomdef room_version = "2.2.5"implementation "androidx.room:room-runtime:$room_version"implementation "androidx.room:room-ktx:$room_version"kapt "androidx.room:room-compiler:$room_version"//Navigationdef nav_version = "2.3.0"implementation "androidx.navigation:navigation-fragment-ktx:$nav_version"implementation "androidx.navigation:navigation-ui-ktx:$nav_version"//Glide 图片处理implementation 'com.github.bumptech.glide:glide:4.11.0'kapt 'com.github.bumptech.glide:compiler:4.11.0'//Timberimplementation '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 (@PrimaryKeyval _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<Image>)

3.2 网络获取数据

配置Retrofit

在这里插入图片描述

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

在这里插入图片描述

获取数据并进行封装为Resource逻辑,Response中数据->Resource由于获取的图片网址为http协议,需要换成https不然无法跳转
protected suspend fun <T> getResult(call: suspend () -> Response<T>): Resource<T> {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 */@Daointerface ImageDao {@Query("SELECT * FROM image_data_table ORDER BY views DESC")fun getAllImages() : LiveData<List<Image>>@Query("SELECT * FROM image_data_table WHERE _id = :id")fun getImage(id: String): LiveData<Image>@Insert(onConflict = OnConflictStrategy.REPLACE)suspend fun insertAll(image_data_table: List<Image>)@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: ImageDaocompanion object {@Volatile private var instance: ImageDatabase? = nullfun 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<T>, networkCall: suspend () -> Resource<A>, saveCallResult: suspend (A) -> Unit): LiveData<Resource<T>> =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在以下类型上使用:

ActivityFragmentViewServiceBroadcastReceiver

在这里插入图片描述

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绑定到navigationnavigation实现两个界面的绑定通过一个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"><fragmentandroid:id="@+id/imageFragment"android:name="com.ffzs.imageapp.ui.images.ImageFragment"android:label="Images"tools:layout="@layout/image_fragment" ><actionandroid:id="@+id/action_imageFragment_to_imageDetailFragment"app:destination="@id/imageDetailFragment" /></fragment><fragmentandroid:id="@+id/imageDetailFragment"android:name="com.ffzs.imageapp.ui.imagesDetail.ImageDetailFragment"android:label="Image Detail"tools:layout="@layout/image_detail_fragment" /></navigation>

5.2 点击后触发onClickedImage

通过ImageViewHolder继承View.OnClickListener,点击后触发onClick

在这里插入图片描述

5.2 传入id

在这里插入图片描述

5.3 获取id

在这里插入图片描述

6.debug

6.1 @AndroidEntryPoint to have a value. Did you forget to apply the Gradle Plugin?

在这里插入图片描述

6.2 java.lang.IllegalStateException: A migration from 1 to 2 is necessary. Please provide a Migration in the builder or call fallbackToDestructiveMigration in the builder in which case Room will re-create all of the tables.

在这里插入图片描述

7.源码

github

郑重声明:本文版权归原作者所有,转载文章仅为传播更多信息之目的,如作者信息标记有误,请第一时间联系我们修改或删除,多谢。