OkHttp初探2:如何使用OkHttp进行下载封装?带进度条?Kotlin+Flow版本。

println(“ n o w T i m e [ nowTime [ nowTime[{Thread.currentThread().name}] ${msg.joinToString(” “)}”)

}

/**

  • 进度通用回调 不使用flow封装的话 使用这个

*/

internal typealias ProgressBlock = (state: DownloadState) -> Unit

/**

  • 下载状态机

*/

sealed class DownloadState {

/**

  • 未开始

*/

object UnStart : DownloadState()

/**

  • 下载中

*/

class Progress(var totalNum: Long, var current: Long) : DownloadState()

/**

  • 下载完成

*/

class Complete(val file: File?) : DownloadState()

/**

  • 下载失败

*/

class Failure(val e: Throwable?) : DownloadState()

/**

  • 下载失败

*/

class FileExistsNoDownload(val file: File?) : DownloadState()

}

下载文件,带进度,一般封装


fun downloadFile(url: String, destFileDirName: String, progressBlock: ProgressBlock) {

//下载状态 默认未开始

var state: DownloadState = DownloadState.UnStart

progressBlock(state)

// TODO: 2021/12/27 file 创建与判断可以封装

/**

  • file 创建判断 可以封装

*/

val file = File(destFileDirName)

val parentFile = file.parentFile

if (!parentFile.exists()) {

parentFile.mkdirs()

}

if (file.exists()) {

//文件存在 不需要下载

state = DownloadState.FileExistsNoDownload(file)

progressBlock(state)

return

} else {

file.createNewFile()

}

//下载

val okHttpClient = OkHttpClient()

val request = Request.Builder().url(url).build()

okHttpClient.newCall(request).enqueue(object : Callback {

override fun onFailure(call: Call, e: IOException) {

state = DownloadState.Failure(e)

progressBlock(state)

}

override fun onResponse(call: Call, response: Response) {

response.use { res ->

//完整长度

var totalLength = 0L

//写入字节

val bytes = ByteArray(2048)

val fileOutputStream = FileOutputStream(file)

res.body?.also { responseBody ->

totalLength = responseBody.contentLength()

}?.byteStream()?.let { inputStream ->

try {

var currentProgress = 0L

var len = 0

state = DownloadState.Progress(totalLength, currentProgress)

do {

if (len != 0) {

currentProgress += len

fileOutputStream.write(bytes)

}

//状态改变

(state as DownloadState.Progress).current = currentProgress

progressBlock(state)

len = inputStream.read(bytes, 0, bytes.size)

} while (len != -1)

//状态改变完成

state = DownloadState.Complete(file)

progressBlock(state)

} catch (e: Exception) {

state = DownloadState.Failure(e)

progressBlock(state)

} finally {

inputStream.close()

fileOutputStream.close()

}

}

}

}

})

}

使用

downloadFile(

“https://dldir1.qq.com/weixin/Windows/WeChatSetup.exe”,

“download/WeChatSetup.exe”

) { state: DownloadState ->

when (val s = state) {

is DownloadState.Complete -> log(“下载完成 文件路径为 ${s.file?.absoluteFile}”)

is DownloadState.Failure -> log(“下载失败 ${s.e?.message}”)

is DownloadState.FileExistsNoDownload -> log(“已经存在 ${s.file?.absoluteFile}”)

is DownloadState.Progress -> log(“下载中 ${(s.current.toFloat() / s.totalNum) * 100}%”)

DownloadState.UnStart -> log(“下载未开始”)

}

}

使用flow封装


对于上述封装使用起来没有问题,但是如果在android上面要把进度显示出来的话,就需要手动切换到UI线程了。不太方便。既然都用kotlin了,那么为什么不解除协程Flow封装呢?

所以,下面基于Flow的封装就来了。直接切换到Main线程,美滋滋。

知识储备:

Kotlin:Flow 全面详细指南,附带源码解析。

Flow : callbackFlow使用心得,避免踩坑!

/**

  • 使用Flow改造文件下载

  • callbackFlow 可以保证线程的安全 底层是channel

*/

fun downloadFileUseFlow(url: String, destFileDirName: String) = callbackFlow {

var state: DownloadState = DownloadState.UnStart

send(state)

//获取文件对象

val file = File(destFileDirName).also { file ->

val parentFile = file.parentFile

if (!parentFile.exists()) {

parentFile.mkdirs()

}

if (file.exists()) {

state = DownloadState.FileExistsNoDownload(file)

send(state)

//流关闭,返回

close()

return@callbackFlow

} else {

file.createNewFile()

}

}

//下载

val okHttpClient = OkHttpClient().newBuilder()

.dispatcher(dispatcher)

.writeTimeout(30, TimeUnit.MINUTES)

.readTimeout(30, TimeUnit.MINUTES)

.build()

val request = Request.Builder()

.url(url)

.build()

okHttpClient.newCall(request).enqueue(object : Callback {

override fun onFailure(call: Call, e: IOException) {

//更新状态

state = DownloadState.Failure(e)

this@callbackFlow.trySendBlocking(state)

close()

}

override fun onResponse(call: Call, response: Response) {

//下载

val body = response.body

if (response.isSuccessful && body != null) {

//完整长度

val totalNum: Long = body.contentLength()

//当前下载的长度

var currentProgress: Long = 0L

var len = 0

response.use {

//等效于 FileOutputStream(file) 输出流

val outputStream = file.outputStream()

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

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

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

img

img

img

img

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

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

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后:学习总结——Android框架体系架构知识脑图(纯手绘xmind文档)

学完之后,若是想验收效果如何,其实最好的方法就是可自己去总结一下。比如我就会在学习完一个东西之后自己去手绘一份xmind文件的知识梳理大纲脑图,这样也可方便后续的复习,且都是自己的理解,相信随便瞟几眼就能迅速过完整个知识,脑补回来。

下方即为我手绘的Android框架体系架构知识脑图,由于是xmind文件,不好上传,所以小编将其以图片形式导出来传在此处,细节方面不是特别清晰。但可给感兴趣的朋友提供完整的Android框架体系架构知识脑图原件(包括上方的面试解析xmind文档)

除此之外,前文所提及的Alibaba珍藏版 Android框架体系架构 手写文档以及一本 《大话数据结构》 书籍等等相关的学习笔记文档,也皆可分享给认可的朋友!

——感谢大家伙的认可支持,请注意:点赞+点赞+点赞!!!

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

习,且都是自己的理解,相信随便瞟几眼就能迅速过完整个知识,脑补回来。

下方即为我手绘的Android框架体系架构知识脑图,由于是xmind文件,不好上传,所以小编将其以图片形式导出来传在此处,细节方面不是特别清晰。但可给感兴趣的朋友提供完整的Android框架体系架构知识脑图原件(包括上方的面试解析xmind文档)
[外链图片转存中…(img-BKXPcwMQ-1713399199666)]

除此之外,前文所提及的Alibaba珍藏版 Android框架体系架构 手写文档以及一本 《大话数据结构》 书籍等等相关的学习笔记文档,也皆可分享给认可的朋友!

——感谢大家伙的认可支持,请注意:点赞+点赞+点赞!!!

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 14
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值