.build()
fun create(serviceClass: Class): T = getRetrofit().create(serviceClass)
inline fun create(): T = create(T::class.java)
}
这里的代码就很简单了,通过网络地址构建一个Retrofit,然后根据传入的Service去访问接口,这里还有一个内联函数。
下面我们来构建这个服务接口,在此之前先在com.llw.goodnews包下新建一个utils包,包下新建一个Constant的类,里面的代码如下:
object Constant {
/**
- 天行数据Key,请使用自己的Key
*/
const val API_KEY = “d8bc937c366fcd1629e00f19105db258”
/**
- 请求接口成功状态码
*/
const val CODE = 200
/**
- 请求接口成功状态描述
*/
const val SUCCESS = “success”
}
这里就是一个常量类,我们在请求API接口时会用到的一些不变的值就放这里。然后我们在network包下新建一个ApiService接口,代码如下:
interface ApiService {
/**
- 获取新闻数据
*/
@GET(“/ncov/index?key=$API_KEY”)
fun getEpidemicNews(): Call
}
下面我们在network包下新建一个发起请求的NetworkRequest类,代码如下:
object NetworkRequest {
/**
- 创建服务
*/
private val service = ServiceCreator.create(ApiService::class.java)
//通过await()函数将getNews()函数也声明成挂起函数。使用协程
suspend fun getEpidemicNews() = service.getEpidemicNews().await()
/**
- Retrofit网络返回处理
*/
private suspend fun Call.await(): T = suspendCoroutine {
enqueue(object : Callback {
//正常返回
override fun onResponse(call: Call, response: Response) {
val body = response.body()
if (body != null) it.resume(body)
else it.resumeWithException(RuntimeException(“response body is null”))
}
//异常返回
override fun onFailure(call: Call, t: Throwable) {
it.resumeWithException(t)
}
})
}
}
这段代码解释一下:首先我们使用ServiceCreator创建了一个ApiService接口的动态代理对象,然后定义了一个getEpidemicNews函数,调用刚刚在ApiService中定义的getEpidemicNews方法,以发起疫情新闻数据请求。这里简化了Retrofit回调的写法,这里定义了一个await()函数,它是一个挂起函数,我们给它声明了一个泛型T,并将await()函数定义成了Call< T >的扩展函数,这样所有返回值是Call类型的Retrofit网络请求接口都可以直接调用await()函数了。
接着,await()函数中使用了suspendCoroutine函数来挂起当前协程,并且由于扩展函数的原因,我们现在拥有了Call对象的上下文,那么这里就可以直接调用enqueue()方法让Retrofit发起网络请求。
最后我们在存储库中发起数据请求,在com.llw.goodnews下创建一个repository包,包下新建一个BaseRepository,里面的代码如下:
open class BaseRepository {
fun fire(context: CoroutineContext, block: suspend () -> Result) =
liveData(context) {
val result = try {
block()
} catch (e: Exception) {
Result.failure(e)
}
//通知数据变化
emit(result)
}
}
这里的fire()函数,按照liveData()函数的参数接收标准定义的一个高阶函数。在fire()函数的内部会先调用一下liveData()函数,然后在liveData()函数的代码块中统一进行try catch处理,并在try语句中调用传入的Lambda表达式中的代码,最终Lambda表达式的执行结果并调用emit()方法发射出去。
然后我们在repository包下再新建一个EpidemicNewsRepository类,用于请求疫情新闻数据,继承自BaseRepository,里面的代码如下:
object EpidemicNewsRepository : BaseRepository() {
fun getEpidemicNews() = fire(Dispatchers.IO) {
val epidemicNews = NetworkRequest.getEpidemicNews()
if (epidemicNews.code == CODE) Result.success(epidemicNews)
else Result.failure(RuntimeException(“getNews response code is ${epidemicNews.code} msg is ${epidemicNews.msg}”))
}
}
这里我们调用父类的fire()函数,将liveData()函数的线程参数类型指定成了Dispatchers.IO,这样的代码块中的所有代码都是运行在子线程中,如果请求状态码是200,则表示成功,那么就使用Kotlin内置的Result.success()方法来包装获取的疫情新闻数据,然后就调用Result.failure()方法来包装一个异常信息。
那么到这里为止,网络框架就搭建完成了,要使用的话还需要一些配置:
这里我们在com.llw.goodnews包下自定义一个App类,继承自Application,代码如下:
class App : Application() {
companion object {
@SuppressLint(“StaticFieldLeak”)
lateinit var context: Context
}
override fun onCreate() {
super.onCreate()
context = applicationContext
}
}
然后因为我们访问的API是http开头的,在Android9.0及以上版本中默认访问https,因此我们需要打开对http的网络访问,在res文件夹下新建一个xml文件夹,在xml文件夹下创建一个network_config.xml,里面的代码如下:
<?xml version="1.0" encoding="utf-8"?>然后我们在AndroidManifest.xml中去配置,如下图所示:
现在万事具备,只差请求了。
进入到MainActivity,新增如下代码:
EpidemicNewsRepository.getEpidemicNews().observe(this@MainActivity) { result ->
val epidemicNews = result.getOrNull()
if (epidemicNews != null) {
Log.d(“TAG”, “onCreate: ${epidemicNews.code}”)
Log.d(“TAG”, “onCreate: ${epidemicNews.msg}”)
Log.d(“TAG”, “onCreate: ${epidemicNews.newslist?.get(0)?.news?.get(0)?.title}”)
Log.d(“TAG”, “onCreate: ${epidemicNews.newslist?.get(0)?.news?.get(0)?.summary}”)
} else {
Log.e(“TAG”, “onCreate: null”)
}
}
添加的位置为下图所示:
这里就是通过请求返回数据,然后打印一下数据,下面来运行一下:
OK,网络框架就没有啥问题了,主要有一个点不爽,就是这里的bean里面太多类,我们写到一个类里面,修改一下EpidemicNews.kt,里面的代码如下:
data class EpidemicNews(val msg: String = “”,
val code: Int = 0,
val newslist: List?)
data class NewslistItem(val news: List?,
val desc: Desc,
val riskarea: Riskarea)
data class NewsItem(val summary: String = “”,
val sourceUrl: String = “”,
val id: Int = 0,
val title: String = “”,
val pubDate: Long = 0,
val pubDateStr: String = “”,
val infoSource: String = “”)
data class Desc(val curedCount: Int = 0,
val seriousCount: Int = 0,
val currentConfirmedIncr: Int = 0,
val midDangerCount: Int = 0,
val suspectedIncr: Int = 0,
val seriousIncr: Int = 0,
val confirmedIncr: Int = 0,
val globalStatistics: GlobalStatistics,
val deadIncr: Int = 0,
val suspectedCount: Int = 0,
val currentConfirmedCount: Int = 0,
val confirmedCount: Int = 0,
val modifyTime: Long = 0,
val createTime: Long = 0,
val curedIncr: Int = 0,
val yesterdaySuspectedCountIncr: Int = 0,
val foreignStatistics: ForeignStatistics,
val highDangerCount: Int = 0,
val id: Int = 0,
val deadCount: Int = 0,
val yesterdayConfirmedCountIncr: Int = 0)
data class Riskarea(val high: List?,
val mid: List?)
data class GlobalStatistics(val currentConfirmedCount: Int = 0,
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)
![](https://i-blog.csdnimg.cn/blog_migrate/efeb913244ca890ab6371187c5fe2406.jpeg)
尾声
对于很多初中级Android工程师而言,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。 整理的这些架构技术希望对Android开发的朋友们有所参考以及少走弯路,本文的重点是你有没有收获与成长,其余的都不重要,希望读者们能谨记这一点。
最后想要拿高薪实现技术提升薪水得到质的飞跃。最快捷的方式,就是有人可以带着你一起分析,这样学习起来最为高效,所以为了大家能够顺利进阶中高级、架构师,我特地为大家准备了一套高手学习的源码和框架视频等精品Android架构师教程,保证你学了以后保证薪资上升一个台阶。
- 思维脑图
- 性能优化学习笔记
- 性能优化视频
当你有了学习线路,学习哪些内容,也知道以后的路怎么走了,理论看多了总要实践的。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
备了一套高手学习的源码和框架视频等精品Android架构师教程,保证你学了以后保证薪资上升一个台阶。
- 思维脑图
[外链图片转存中…(img-uoCMpT1R-1712138715394)] - 性能优化学习笔记
[外链图片转存中…(img-J1yrOXUC-1712138715394)]
[外链图片转存中…(img-gpA5Wx7K-1712138715395)]
[外链图片转存中…(img-BvAz2DGw-1712138715395)]
[外链图片转存中…(img-z2QkPFHA-1712138715395)]
- 性能优化视频
[外链图片转存中…(img-XCrQKFsf-1712138715396)]
当你有了学习线路,学习哪些内容,也知道以后的路怎么走了,理论看多了总要实践的。