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移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

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

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

[外链图片转存中…(img-vDLFZifz-1715713090382)]

[外链图片转存中…(img-AqEFZfrK-1715713090383)]

[外链图片转存中…(img-9yoOTTiF-1715713090383)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门

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

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值