总结
这次面试问的还是还是有难度的,要求当场写代码并且运行,也是很考察面试者写代码
因为Android知识体系比较庞大和复杂的,涉及到计算机知识领域的方方面面。在这里我和身边一些朋友特意整理了一份快速进阶为Android高级工程师的系统且全面的学习资料。涵盖了Android初级——Android高级架构师进阶必备的一些学习技能。
附上:我们之前因为秋招收集的二十套一二线互联网公司Android面试真题(含BAT、小米、华为、美团、滴滴)和我自己整理Android复习笔记(包含Android基础知识点、Android扩展知识点、Android源码解析、设计模式汇总、Gradle知识点、常见算法题汇总。)
里面包含不同方向的自学编程路线、面试题集合/面经、及系列技术文章等,资源持续更新中…
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
- 结论
async如果加上了await就会阻塞线程,
runningBlock会阻塞所在线程
- 代码
fun main() {
log(“A–准备进入runningBlock”) //A
runBlocking {
log(“B–进入runningBlock” )//B
delay(1)
log(“C–执行完成runningBLock”)//C
}
log(“D–runBlocking代码块外执行”)//D
runBlocking {
delay(3)
}
}
- 执行结果
日志: A–准备进入runningBlock
日志: B–进入runningBlock
日志: C–执行完成runningBLock
日志: D–runBlocking代码块外执行
- 结论
本示例中runBlocking代码块中的B和C顺序执行完成后才执行的代码块外面的D,这证明了runningBlock会阻塞线程。
suspend函数会阻塞协程
suspend会阻塞所在的协程,不会阻塞线程,因为suspend无法在线程内部调用
- 代码
fun main() = runBlocking {
log(“A–调用suspend函数前”)
wait2Second()
log(“B–调用suspend函数后”)
delay(3)
}
suspend fun wait2Second() {
log(“C–调用suspend函数开始”)
delay(2)
log(“D–调用suspend函数结束”)
}
- 日志
日志: A–调用suspend函数前
日志: C–调用suspend函数开始
日志: D–调用suspend函数结束
日志: B–调用suspend函数后
- 结论
suspend函数会阻塞协程
withContext是否会阻塞协程
- 代码
fun main() = runBlocking {
log(“A–执行withContext前”)
withContext(Dispatchers.Default){
log(“B–开始执行withContext”)
delay(1)
log(“C–withContext执行完成”)
}
log(“D–执行withContext后”)
delay(2)
}
- 日志
日志: A–执行withContext前
日志: B–开始执行withContext
日志: C–withContext执行完成
日志: D–执行withContext后
- 结论
withContext会阻塞协程,withContext只能运行在协程,不能运行在线程中。本例中的Default改为Main就会报错,因为Main相当于运行在主线程
等待一个作业
launch方法会返回一个Job对象,Job对象调用了join方法后,会等待launch协程体执行完成再继续后面的任务。
join方法只能在协程体里面调用,不能在线程中调用
- 代码
fun main() = runBlocking {
log(“A”)
val job = GlobalScope.launch {
log(“B”)
delay(2)
log(“C”)
}
log(“D”)
job.join()
log(E)
}
- 日志
日志: A
日志: D
日志: B
日志: C
日志: E
- 结论
join会等待协程体执行完成
结构化并发
使用GlobalScope.launch是很危险的,因为它创建的是一个顶层协程,如果launch方法里执行了耗时任务,而我们的对象过早的被回收就会发生内存泄露。
这种情况的解决办法就是在我们指定的作用域范围内开启协程
举例:
- 代码
fun main() = runBlocking {
log(A)
//这个launch和GlobalScope.launch是不同的,它是运行在runBlocking的作用域内的
launch {
log(B)
delay(2)
log©
}
log(D)
}
- 日志
日志: A
日志: D
日志: B
日志: C
- 因为launch方法运行在外部runBlocking作用域内,
作用域构建器(直接抄官方)
除了由不同的构建器提供协程作用域之外,还可以使用 coroutineScope 构建器声明自己的作用域。它会创建一个协程作用域并且在所有已启动子协程执行完毕之前不会结束。
runBlocking 与 coroutineScope 可能看起来很类似,因为它们都会等待其协程体以及所有子协程结束。 主要区别在于,runBlocking 方法会阻塞当前线程来等待, 而 coroutineScope 只是挂起,会释放底层线程用于其他用途。 由于存在这点差异,runBlocking 是常规函数,而 coroutineScope 是挂起函数
- 代码
fun main() = runBlocking { // this: CoroutineScope
launch {
delay(200L)
println(“Task from runBlocking”)
}
coroutineScope { // 创建一个协程作用域
launch {
delay(500L)
println(“Task from nested launch”)
}
delay(100L)
println(“Task from coroutine scope”) // 这一行会在内嵌 launch 之前输出
}
println(“Coroutine scope is over”) // 这一行在内嵌 launch 执行完毕后才输出
}
- 输出
学习福利
【Android 详细知识点思维脑图(技能树)】
其实Android开发的知识点就那么多,面试问来问去还是那么点东西。所以面试没有其他的诀窍,只看你对这些知识点准备的充分程度。so,出去面试时先看看自己复习到了哪个阶段就好。
虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。
这里附上上述的面试题相关的几十套字节跳动,京东,小米,腾讯、头条、阿里、美团等公司19年的面试题。把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。
由于篇幅有限,这里以图片的形式给大家展示一小部分。
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。
由于篇幅有限,这里以图片的形式给大家展示一小部分。
[外链图片转存中…(img-C2aOBQ1d-1715894737222)]
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!