最后
我见过很多技术leader在面试的时候,遇到处于迷茫期的大龄程序员,比面试官年龄都大。这些人有一些共同特征:可能工作了5、6年,还是每天重复给业务部门写代码,工作内容的重复性比较高,没有什么技术含量的工作。问到这些人的职业规划时,他们也没有太多想法。
其实30岁到40岁是一个人职业发展的黄金阶段,一定要在业务范围内的扩张,技术广度和深度提升上有自己的计划,才有助于在职业发展上有持续的发展路径,而不至于停滞不前。
不断奔跑,你就知道学习的意义所在!
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
在我们日常处理一些耗时任务的时候,有很多的方案,比如
1.可以通过Handler().sendMessageDelayed() 达到延迟加载。
原理:将消息加入队列中,然后MessageQueue会根据延时的时间进行队列的排序,时间最短的在前,如果没有要执行的,就进行阻塞,阻塞的时间为最先要执行的任务的等待时间,如果不再添加新任务,则等时间到了会自动执行,如果添加了新任务,则重新排序,然后唤醒当前线程,将排序后,最先要执行的等待时间进行阻塞或者直接执行。
缺点:但是项目中是不建议这样用的,因为会强占CPU,性能会进行耗损,比如一个页面的一些第三方服务进行初始化操作,虽然说是可以延迟一段时间再去初始化,但是如果该页面一直在执行,比如有个定时器或者轮询请求接口等,那么到了时间,依然是要强占CPU来执行我们的第三方服务的初始化操作。所以不能直接这么用
不知道会不会有杠精,“我们平时也是这么用的呀,也没问题呀”。但是你要记住,我聊的是大型项目,比如中石油终端,一个APP中,不光要作为主设备接收其他设备传递的数据,保持的长链接,还通过自定义的一些协议,比如FTFS协议,与硬件进行连接,如加油机,前庭控制器,液位仪等,你直接来个延迟初始化,一开始没什么,等延迟时间到了,如果人员也在操作,油机也在实时上报数据,直接卡死你。
2.IldeHandler
这个的确能解决我们之前尴尬的问题,它的主张是在CPU空闲时再进行操作,不抢占CPU
同学们,面经是不是就只写到这呀,那你们考虑过,如果请求过多尼,如果并发尼?如果空闲执行中执行的任务还必须有先后执行的顺序尼。比如A页面,B页面都把自己耗时的方法加入到了空闲执行队列里面,但是要想执行B页面耗时方法,必须得先执行页面A中的方法,你该怎么做?
废话不多说,直接上代码,顺便附一张之前战斗过的地方,项目虽好,但是工作室太“简陋”,做完几个版本就溜了。。
3.聊一聊IdleHandler的优化及封装
不知道task是啥的,就去看第三章内容。
/**
- @author: lybj
- @date: 2020/5/26
- @Description: 空闲队列
*/
public class IldeTaskManager {
private Queue ildeTaskQueue = new LinkedList<>();
private MessageQueue.IdleHandler idleHandler = new MessageQueue.IdleHandler(){
@Override
public boolean queueIdle() {
if(ildeTaskQueue.size() > 0){
// 如果CPU空闲了,
Task idleTask = ildeTaskQueue.poll();
new TaskRunnable(idleTask).run();
}
// 如果返回false,则移除该 IldeHandler
return !ildeTaskQueue.isEmpty();
}
};
public IldeTaskManager addTask(Task task){
ildeTaskQueue.add(task);
return this;
}
/**
- 执行空闲方法,因为用了DispatchRunnable,所以会优先处理需要依赖的task,再处理本次需要处理的task,顺序执行
- */
public void start(){
Looper.myQueue().addIdleHandler(idleHandler);
}
}
调用的话也很简单
new IldeTaskManager()
.addTask(new InitBaiduMapTask())
.addTask(new InitBuglyTask())
.start();
4.其他代码
不明白的,去看上一章的讲解,这章本来就是在上一章内容上增加的拓展
TaskRunnable
public class TaskRunnable implements Runnable {
private Task task;
private TaskManager taskManager;
public TaskRunnable(Task task) {
this.task = task;
}
public TaskRunnable(Task task, TaskManager taskManager) {
this.task = task;
this.taskManager = taskManager;
}
@Override
public void run() {
TraceCompat.beginSection(task.getClass().getSimpleName());
Process.setThreadPriority(task.priority());
task.startLock();
task.run();
// 执行Task的尾部任务
Runnable tailRunnable = task.getTailRunnable();
if (tailRunnable != null) {
tailRunnable.run();
}
if (!task.runOnMainThread()) {
if(taskManager != null){
taskManager.unLockForChildren(task);
taskManager.finish(task);
}
}
TraceCompat.endSection();
}
}
task
public abstract class Task implements ITask {
// 当前Task依赖的Task数量(需要等待被依赖的Task执行完毕才能执行自己),默认没有依赖
private CountDownLatch taskCountDownLatch = new CountDownLatch(dependentArr() == null ? 0 : dependentArr().size());
/**
- 当前Task等待,让依赖的Task先执行
*/
@Override
public void startLock() {
try {
taskCountDownLatch.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
/**
- 依赖的Task执行完一个
*/
@Override
public void unlock() {
taskCountDownLatch.countDown();
}
/**
- Task的优先级,运行在主线程则不要去改优先级
*/
@Override
小福利:
在当下这个碎片化信息环境的时代,很多资源都可以在网络上找到,只取决于你愿不愿意找或是找的方法对不对了
很多朋友不是没有资料,大多都是有几十上百个G,但是杂乱无章,不知道怎么看从哪看起,甚至是看后就忘
如果大家觉得自己在网上找的资料非常杂乱、不成体系的话,我也分享一套给大家,比较系统,我平常自己也会经常研读。
2021大厂最新Android面试真题解析
各个模块学习视频:如数据结构与算法
只有系统,有方向的学习,才能在段时间内迅速提高自己的技术。
这份体系学习笔记,适应人群:**第一,**学习知识比较碎片化,没有合理的学习路线与进阶方向。**第二,**开发几年,不知道如何进阶更进一步,比较迷茫。第三,到了合适的年纪,后续不知道该如何发展,转型管理,还是加强技术研究。如果你有需要,我这里恰好有为什么,不来领取!说不定能改变你现在的状态呢!点赞+评论即可获得!
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!