}
}
})
}
inline fun multipartUpload(
url: String,
filePath: String,
contentType: MediaType? = null,
params: Map<String, String>? = null,
crossinline block: (UploadState) -> Unit
) {
var state: UploadState = UploadState.UnStart
block(state)
val file = File(filePath)
val body = MultipartBody.Builder()
.also {
params?.forEach { (k, v) ->
it.addFormDataPart(k, v)
}
}.also {
if (file.exists()) {
it.addFormDataPart(“filename”, file.name, file.asRequestBody(contentType))
}
}.build()
val request = Request.Builder()
.url(url)
.post(body)
// .addHeader() 可以增加请求头
.build()
val client = OkHttpClient.Builder()
.dispatcher(dispatcher)
.writeTimeout(30, TimeUnit.MINUTES)
.readTimeout(30, TimeUnit.MINUTES)
.connectTimeout(70, TimeUnit.SECONDS)
.build()
client.newCall(request).enqueue(object : Callback {
override fun onFailure(call: Call, e: IOException) {
log(e.message)
}
override fun onResponse(call: Call, response: Response) {
if (response.isSuccessful) {
state = UploadState.Complete
block(state)
} else {
log(“请求失败”)
}
}
})
}
fun multipartUploadProgress(
url: String,
filePath: String,
contentType: MediaType? = null,
params: Map<String, String>? = null,
block: (UploadState) -> Unit
) {
var state: UploadState = UploadState.UnStart
block(state)
val file = File(filePath)
val body = MultipartBody.Builder()
.also {
params?.forEach { (k, v) ->
it.addFormDataPart(k, v)
}
}.also {
if (file.exists()) {
it.addFormDataPart(“filename”, file.name, file.asProgressRequestBody(contentType, block))
}
}.build()
val request = Request.Builder()
.url(url)
.post(body)
// .addHeader() 可以增加请求头
.build()
val client = OkHttpClient.Builder()
.dispatcher(dispatcher)
.writeTimeout(30, TimeUnit.MINUTES)
.readTimeout(30, TimeUnit.MINUTES)
.connectTimeout(70, TimeUnit.SECONDS)
.build()
client.newCall(request).enqueue(object : Callback {
override fun onFailure(call: Call, e: IOException) {
log(e.message)
}
override fun onResponse(call: Call, response: Response) {
if (response.isSuccessful) {
state = UploadState.Complete
block(state)
} else {
log(“请求失败”)
}
}
})
}
/**
- 带进度条上传的功能
*/
private fun File.asProgressRequestBody(contentType: MediaType? = null, block: (UploadState) -> Unit?): RequestBody {
return object : RequestBody() {
override fun contentType() = contentType
override fun contentLength() = length()
override fun writeTo(sink: BufferedSink) {
source().use { source ->
val buffer = Buffer()
var readCount = 0L
var progress = 0L
val progressBlock = UploadState.Progress(contentLength(), progress)
try {
do {
if (readCount != 0L) {
progress += readCount
progressBlock.current = progress
sink.write(buffer, readCount)
block(progressBlock)
}
readCount = source.read(buffer, 2048)
} while (readCount != -1L)
} catch (e: Exception) {
// e.printStackTrace()
block(UploadState.Error(e))
}
}
}
}
}
使用
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新
如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
总结
最后为了帮助大家深刻理解Android相关知识点的原理以及面试相关知识,这里放上相关的我搜集整理的24套腾讯、字节跳动、阿里、百度2019-2021面试真题解析,我把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包知识脉络 + 诸多细节。
还有 高级架构技术进阶脑图、Android开发面试专题资料 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
2021年虽然路途坎坷,都在说Android要没落,但是,不要慌,做自己的计划,学自己的习,竞争无处不在,每个行业都是如此。相信自己,没有做不到的,只有想不到的。祝大家2021年万事大吉。
一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
brq1ama-1712661473510)]
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
2021年虽然路途坎坷,都在说Android要没落,但是,不要慌,做自己的计划,学自己的习,竞争无处不在,每个行业都是如此。相信自己,没有做不到的,只有想不到的。祝大家2021年万事大吉。
一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
[外链图片转存中…(img-eZI4To76-1712661473511)]