Java定时器(实现每月1号、每日、每15分钟自动执行任务)

使用注解的形式,跑起代码

一、代码

1、思路:在tomcat中,添加监听器,在监听器中设置定时任务。
2、监听:新建监听类implents ServletContextListener,实现其中的方法即可。

讲解

  1. 创建一个基准时间defaultdate(每日8点),用于参照,在此时间以后的多长周期内执行操作。

  2. schedule(task, firstTime, period); 方法参数介绍:

    task:TimerTask任务,用内部匿名类的方式新建一个即可(当然也可以在外部类中建一个类,用于写任务,写法麻烦点),实现run()方法,在Run中写你要执行操作即可。

    firstTime:任务首次执行时间。当系统时间大于firstTime,会立即执行一次任务。当系统时间小于firstTime,则等到时间等于firstTime时才执行。所以用schedule实现定时任务,最重要的的就是控制这个firstTime。

    period:执行周期,单位:毫秒。一天写法:24 * 60 * 60 * 1000

  3. 如何判断是每月1号?用Calendar的Calendar.DAY_OF_MONTH,每月首天
    返回值为1。每天都执行下判断,到每月一号时,即可实现每月执行一次

Java监听器代码:

package com.nxt.myprofile.monitor;

import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import javax.servlet.annotation.WebListener;
import java.util.Calendar;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;

/**
 * 别人的博客:https://blog.csdn.net/x541211190/article/details/64125781?ops_request_misc=%257B%2522request%255Fid%2522%253A%2522162544827316780271550968%2522%252C%2522scm%2522%253A%252220140713.130102334.pc%255Fall.%2522%257D&request_id=162544827316780271550968&biz_id=0&utm_medium=distribute.pc_search_result.none-task-blog-2~all~first_rank_v2~rank_v29-1-64125781.first_rank_v2_pc_rank_v29_1&utm_term=Java%E5%AE%9A%E6%97%B6%E5%99%A8%EF%BC%88%E5%AE%9E%E7%8E%B0%E6%AF%8F%E6%9C%881%E5%8F%B7%E3%80%81%E6%AF%8F%E6%97%A5%E3%80%81%E6%AF%8F15%E5%88%86%E9%92%9F%E8%87%AA%E5%8A%A8%E6%89%A7%E8%A1%8C%E4%BB%BB%E5%8A%A1%EF%BC%89&spm=1018.2226.3001.4187l
 * @author LunarYouI
 * @create 2021-07-03 17:44
 */
@WebListener
public class SendWsListener implements ServletContextListener {
    @Override
    public void contextDestroyed(ServletContextEvent arg0) {
        System.out.println("定时发送Xml信息监听--已关闭!");
    }

    @Override
    public void contextInitialized(ServletContextEvent arg0) {
        Calendar calendar = Calendar.getInstance();
        int year = calendar.get(Calendar.YEAR);
        int month = calendar.get(Calendar.MONTH);
        //设置24小时制度,Calendar.HOUR是12小时制度
        int day = calendar.get(Calendar.DAY_OF_MONTH);
        calendar.set(year, month, day, 8, 00, 00);
        // 当天8点(默认执行时间)
        Date defaultdate = calendar.getTime();
        Date sendDate = null;
        if (defaultdate.before(new Date())) {
            // 若当前时间超过了defaultdate时间,当天不再执行,则将执行时间sendDate改为明天8点
            calendar.add(Calendar.DATE, 1);
            sendDate = calendar.getTime();
        }else {
            // 若当前时间没有超过defaultdate时间,则将执行时间sendDate改为defaultdate
            sendDate = defaultdate;
        }


        /**
         * ----------------每刻任务 ----------------
         * 启动服务器后,若此时时间没过8点,等待。到了8点自动执行一次,15分钟后再执行一次,周而复始
         * 启动服务器后,若此时时间超过8点,会立刻执行一次,等到15分钟后再次执行一次,周而复始 到了第二天,不会再判断是否是8点,这个开始时间,只会判断一次而已
         */
        Timer qTimer = new Timer();
        qTimer.schedule(new TimerTask() {

            @Override
            public void run() {
                System.out.println("每刻任务已执行");
                // TODO 写你的逻辑
            }
        }, defaultdate, 15 * 60 * 1000);// 定时每15分钟
        System.out.println("每刻定时发送Xml信息监听--已启动!");

        /**
         * ----------------每日任务 ----------------
         * 启动服务器后,若此时时间没过8点,等待。到了8点自动执行一次,24小时后(第二天8点)再执行一次,周而复始
         * 启动服务器后,若此时时间已经超过8点,则等到24小时后(第二天8点)才执行一次,周而复始
         */
        Timer dTimer = new Timer();
        dTimer.schedule(new TimerTask() {

            @Override
            public void run() {
                System.out.println("每日任务已经执行");
                // TODO 写你的逻辑
            }
        }, sendDate, 24 * 60 * 60 * 1000);// 定时24小时:24 * 60 * 60 * 1000
        System.out.println("每日定时发送Xml信息监听--已启动!");

        /**
         * ----------------每月任务 ----------------
         * 启动服务器后,若此时时间没过8点,等待。到了8点自动执行判断是否是当前月份的1号,若是则执行一次,
         * 24小时后(第二天8点)再执行一次判断(每月1号以后后的29天或30天后才会是下月1号,再执行一次),周而复始 启动服务器后,若此时时间已经超过8点,会立刻执行一次,等到下个月1号再次执行一次,周而复始
         */
        Timer mTimer = new Timer();
        mTimer.schedule(new TimerTask() {

            @Override
            public void run() {
                Calendar c = Calendar.getInstance();
                int day = c.get(Calendar.DAY_OF_MONTH);
                System.out.println("月任务 判断中");
                if (day == 1) {
                    // 每天执行,若为每月1号才执行
                    System.out.println("月任务执行已执行");
                    // TODO 写你的逻辑
                }

            }
        }, sendDate, 24 * 60 * 60 * 1000);// 每天执行一次检查

        System.out.println("每月定时发送Xml信息监听--已启动!");

    }
}

注意:

  • 上面代码中需要加注解@WebListener;

  • 启动类需要加注解:@ServletComponentScan(“com.nxt.myprofile.monitor”)
    在这里插入图片描述

  • 拿每日任务来说:

    程序启动的系统时间若在8点之前,则等到8点执行任务;若系统时间在8点之后,则将sendDate修改为明天8点,则定时任务会在明天8天执行,今天将不再执行
    如果想让代码今天也执行一次可以将sendDate修改成defaultdate
    在这里插入图片描述

    在这里插入图片描述