fix cron bug

This commit is contained in:
Looly 2020-04-19 11:41:10 +08:00
parent f4fc97f9de
commit 8e42898d46
4 changed files with 84 additions and 20 deletions

View File

@ -7,8 +7,11 @@
### 新特性 ### 新特性
* 【core 】 增加NetUtil.isOpen方法 * 【core 】 增加NetUtil.isOpen方法
* 【core 】 增加ThreadUtil.sleep和safeSleep的重载
### Bug修复 ### Bug修复
* 【db 】 修复PageResult.isLast计算问题 * 【db 】 修复PageResult.isLast计算问题
* 【cron 】 修复更改系统时间后CronTimer被阻塞的问题issue#838@Github
------------------------------------------------------------------------------------------------------------- -------------------------------------------------------------------------------------------------------------
## 5.3.1 (2020-04-17) ## 5.3.1 (2020-04-17)

View File

@ -226,11 +226,23 @@ public class ThreadUtil {
if (millis == null) { if (millis == null) {
return true; return true;
} }
return sleep(millis.longValue());
}
try { /**
Thread.sleep(millis.longValue()); * 挂起当前线程
} catch (InterruptedException e) { *
return false; * @param millis 挂起的毫秒数
* @return 被中断返回false否则true
* @since 5.3.2
*/
public static boolean sleep(long millis) {
if (millis > 0) {
try {
Thread.sleep(millis);
} catch (InterruptedException e) {
return false;
}
} }
return true; return true;
} }
@ -243,15 +255,36 @@ public class ThreadUtil {
* @see ThreadUtil#sleep(Number) * @see ThreadUtil#sleep(Number)
*/ */
public static boolean safeSleep(Number millis) { public static boolean safeSleep(Number millis) {
long millisLong = millis.longValue(); if (millis == null) {
return true;
}
return safeSleep(millis.longValue());
}
/**
* 考虑{@link Thread#sleep(long)}方法有可能时间不足给定毫秒数此方法保证sleep时间不小于给定的毫秒数
*
* @param millis 给定的sleep时间
* @return 被中断返回false否则true
* @see ThreadUtil#sleep(Number)
* @since 5.3.2
*/
public static boolean safeSleep(long millis) {
long done = 0; long done = 0;
while (done < millisLong) { long before;
long before = System.currentTimeMillis(); long spendTime;
if (false == sleep(millisLong - done)) { while (done >= 0 && done < millis) {
before = System.currentTimeMillis();
if (false == sleep(millis - done)) {
return false; return false;
} }
long after = System.currentTimeMillis(); spendTime = System.currentTimeMillis() - before;
done += (after - before); if (spendTime <= 0) {
// Sleep花费时间为0或者负数说明系统时间被拨动
break;
}
done += spendTime;
} }
return true; return true;
} }
@ -318,6 +351,13 @@ public class ThreadUtil {
} }
} }
/**
* 等待当前线程结束. 调用 {@link Thread#join()} 并忽略 {@link InterruptedException}
*/
public static void waitForDie() {
waitForDie(Thread.currentThread());
}
/** /**
* 等待线程结束. 调用 {@link Thread#join()} 并忽略 {@link InterruptedException} * 等待线程结束. 调用 {@link Thread#join()} 并忽略 {@link InterruptedException}
* *

View File

@ -44,16 +44,18 @@ public class CronTimer extends Thread implements Serializable {
long sleep; long sleep;
while(false == isStop){ while(false == isStop){
//下一时间计算是按照上一个执行点开始时间计算的 //下一时间计算是按照上一个执行点开始时间计算的
//此处除以定时单位是为了清零单位以下部分例如单位是分则秒和毫秒清零
nextTime = ((thisTime / timerUnit) + 1) * timerUnit; nextTime = ((thisTime / timerUnit) + 1) * timerUnit;
sleep = nextTime - System.currentTimeMillis(); sleep = nextTime - System.currentTimeMillis();
if (sleep > 0 && false == ThreadUtil.safeSleep(sleep)) { if(isValidSleepMillis(sleep, timerUnit)){
//等待直到下一个时间点如果被中断直接退出Timer if (false == ThreadUtil.safeSleep(sleep)) {
break; //等待直到下一个时间点如果被中断直接退出Timer
break;
}
//执行点时间记录为执行开始的时间而非结束时间
thisTime = System.currentTimeMillis();
spawnLauncher(thisTime);
} }
//执行点时间记录为执行开始的时间而非结束时间
thisTime = System.currentTimeMillis();
spawnLauncher(thisTime);
} }
log.debug("Hutool-cron timer stoped."); log.debug("Hutool-cron timer stoped.");
} }
@ -73,4 +75,22 @@ public class CronTimer extends Thread implements Serializable {
private void spawnLauncher(final long millis){ private void spawnLauncher(final long millis){
this.scheduler.taskLauncherManager.spawnLauncher(millis); this.scheduler.taskLauncherManager.spawnLauncher(millis);
} }
/**
* 检查是否为有效的sleep毫秒数包括
* <pre>
* 1. 是否&gt;0防止用户向未来调整时间
* 1. 是否&lt;两倍的间隔单位防止用户向历史调整时间
* </pre>
*
* @param millis 毫秒数
* @param timerUnit 定时单位为秒或者分的毫秒值
* @return 是否为有效的sleep毫秒数
* @since 5.3.2
*/
private static boolean isValidSleepMillis(long millis, long timerUnit){
return millis > 0 &&
// 防止用户向前调整时间导致的长时间sleep
millis < (2 * timerUnit);
}
} }

View File

@ -36,13 +36,14 @@ public class CronTest {
} }
@Test @Test
@Ignore // @Ignore
public void cronTest2() { public void cronTest2() {
// 支持秒级别定时任务 // 支持秒级别定时任务
CronUtil.setMatchSecond(true); CronUtil.setMatchSecond(true);
CronUtil.start(); CronUtil.start();
ThreadUtil.sleep(30000); ThreadUtil.waitForDie();
Console.log("Exit.");
} }
@Test @Test