scheduledexecutorservice
scheduledexecutorservice的创建就不用多说,直接上代码
executorService = Executors.newSingleThreadScheduledExecutor();
ScheduledExecutorService的停止有shutdown和shutdownNow之分,可以看下它们之间的区别,但多数情况下都不能结束在线程池中的任务。不过有个小技巧可以让真正的逻辑不执行,就是在run方法中添加条件判断。如下:
@Override
public void run() {
if (mFrag.mWiFi != null) { //添加条件判断,在shutdown之前把mWifi置为空
mFrag.search(); //这样虽不能取消run方法的执行,但可以让真正的searchWifi逻辑不执行
}
}
但是如果我们有业务需要,要求定时任务可以取消,再创建,该如何操作呢?
常规的写法就是
executorService.shutdownNow(); //取消
executorService.scheduleatfixedrate(new Executor(),30,30, TimeUnit.SECONDS); //开始
坑1:
这样的话,熟悉的同学就知道会发生RejectedexecutionException异常。因为executorService已经关闭了。如果需要继续开始任务,则需要重新new 对象Executors.newSingleThreadScheduledExecutor()。
坑2:
在坑1的基础上,我们想想会发生什么。坑1取消事件调用的是shutdownNow方法,开始事件是newSingleThreadScheduledExecutor创建对象后,再scheduleAtFixedRate。这样的话,我们从log分析就可以看出来,任务可能多次执行,就是上一次任务并没有销毁。我们每开始一个任务前,就需要把上一次任务关闭、置空。像这样:
private void beginExecutorScan() {
//先取消上一个任务,防止重复的任务
endExecutorScan();
executorService = Executors.newSingleThreadScheduledExecutor();
if (executorService != null) {
Lg.e("ScheduledExecutorService beginExecutorScan");
executorService.scheduleAtFixedRate(new ScanScheduledExecutor(Frag.this),30,30, TimeUnit.SECONDS);
}
}
private void endExecutorScan() {
if(executorService != null) {
Lg.e("ScheduledExecutorService endExecutorScan");
executorService.shutdownNow();
}
executorService = null;//非单例模式,置空防止重复的任务
}
坑3:
如果定时任务执行过程中遇到发生异常,则后面的任务将不再执行。处理方式是在run中加try catch,这样当一个任务发生异常时,后续的任务还会继续执行。
解决完以上3种坑后,完整的可以重复开始、停止的ScheduledExecutorService周期性任务像这样:
private void beginExecutorScan() {
//先取消上一个任务,防止重复的任务
endExecutorScan();
executorService = Executors.newSingleThreadScheduledExecutor();
if (executorService != null) {
Lg.e("ScheduledExecutorService beginExecutorScan");
executorService.scheduleAtFixedRate(new ScanScheduledExecutor(Frag.this),30,30, TimeUnit.SECONDS);
}
}
private void endExecutorScan() {
if(executorService != null) {
Lg.e("ScheduledExecutorService endExecutorScan");
executorService.shutdownNow();
}
executorService = null;//非单例模式,置空防止重复的任务
}
//静态弱引用方式,防止内存泄露
static class ScanScheduledExecutor implements Runnable {
weakreference<Frag> mFragReference;
ScanScheduledExecutor(Frag frag) {
mFragReference = new WeakReference<>(frag);
}
@Override
public void run() {
try {
final Frag mFrag = mFragReference.get();
if (mFrag == null)
return;
if (mFrag.mWifi != null) { //添加条件判断,在shutdown之前把mWifi置为空
mFrag.search(); //这样虽不能取消run方法的执行,但可以让真正的searchWifi逻辑不执行
}
} catch (Exception e) {
Lg.e("ScheduledExecutorService e:"+e.toString());
}
}
}
附:
scheduleAtFixedRate 与 scheduleWithFixedDelay 的区别:
scheduleAtFixedRate ,是以上一个任务开始的时间计时,N秒过去后,检测上一个任务是否执行完毕,如果上一个任务执行完毕,则当前任务立即执行,如果上一个任务没有执行完毕,则需要等上一个任务执行完毕后立即执行。
scheduleWithFixedDelay,是以上一个任务结束时开始计时,N秒过去后,立即执行。
相关阅读
ScheduledExecutorService 延迟/周期执行线程池
目录 Executor 结构图 ScheduleExecutorService 简述 对象创建方式 方法详解 schedule Runnable schedule Callable scheduleAt
ScheduledExecutorService定时周期执行指定的任务
一:简单说明ScheduleExecutorService接口中有四个重要的方法,其中scheduleAtFixedRate和scheduleWithFixedDelay在实现定时程序时比
ScheduledExecutorService:多线程任务调度
今天使用Timer实现任务调度时,阿里巴巴Java开发规范提示 多线程并行处理定时任务时,Timer运行多个TimeTask时,只要其中之一没有捕获