JFinal - scheduler 插件做定时任务

我在项目中遇到一个需求:服务运行期间,数据库要定期去监测某表并且更新。

正好项目是使用 jfinal 做的,于是就用了 jfinal-scheduler 插件来解决(jfinal-scheduler 可以用来做定时任务):

配置插件,在自定义的 JFinalConfig 类中引入此插件:

/**
 * 配置插件
 */
public void configPlugin(Plugins me) {
    // ...
    // 定时器插件
    SchedulerPlugin sp = new SchedulerPlugin();
    Runnable task = new Task();
    sp.fixedRateSchedule(task, 60);
    me.add(sp);
    // ...
}

完善线程

public class Task implements Runnable{
    @Override
    public void run() {
        // DML操作
    }
}

完成上述工作之后,我启动服务器,发现在 DML 操作之处必现线程阻塞。然而我将这段 DML 操作放到 Controller 中是能够正常执行并查出数据的。

后来发现原因在这里:

public class SchedulerPlugin implements IPlugin {

    // ...
    private final ScheduledThreadPoolExecutor taskScheduler;

    public SchedulerPlugin() {
        this(getBestPoolSize(), null);
    }

    public ScheduledFuture<?> fixedRateSchedule(Runnable task, int periodSeconds) {
        return taskScheduler.scheduleAtFixedRate(task, 0, periodSeconds, TimeUnit.SECONDS);
    }

    // ...
}

在插件调用 fixedRateSchedule(...) 的时候,会调 ScheduledThreadPoolExecutor.scheduleAtFixedRate(...),第二个参数的意思是第一次启动线程会在多少秒后执行,因为源码中这个参数是 0,也就是马上执行的意思,然而此时数据库还没有连上呢,查询操作自然就出异常了。我把 0 改成了 60,果然,1分钟后,线程启动,运行正常。

用到的源码和jar包:

myaniu-jfinal-scheduler-master.zip

cron4j-2.2.5.zip

上一篇:Java邮件发送与接收原理


下一篇:c#实现邮件发送链接激活