原创SpringBoot实现月末功能

Posted DCTANT

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了原创SpringBoot实现月末功能相关的知识,希望对你有一定的参考价值。

前言

众所周知的事是Spring的cron表达式和Linux、Quartz的cron表达式是不同的,比如说Spring和Linux的cron表达式星期是从0开始的,星期天是0,星期一是1,以此类推,星期六是6,星期天也是7,也就是说Spring、Linux的cron表达式中的星期天既可以等于0,也可以等于7。

而Quartz中的星期是从1开始的,星期天是1,星期六是7。但是很多在线cron表达式生成的网站并没有说明他们是按照哪个规则是生成的,以至于有个项目要在周六执行,等了好久都没执行,怎么测都不执行,才发现SpringBoot使用@Scheduled注解使用了Quartz表达式了,当然这个表达式是从在线cron表达式生成的网站生成的。

报错记录

结果这次项目中还要实现一个月末统计的功能,然后查阅资料发现cron表达式的月这一栏可以填L,表示Last最后一天的意思,在周和月都可以填这个L以表示这周的最后一天和这月的最后一天。我当时就很高兴啊,省得再去算这个月末了,然后我就直接照搬了,结果呢,Spring的cron表达式是不支持这个L的!!这不坑我呢!!

比如这个cron表达式:0 0 8 L * ?

这个表达式在在线crob表达式生成器中是正常执行的,但是放在SpringBoot的@Scheduled注解中启动就会报错!

 意思就是Spring无法解析这个L,不认识它,也就是说Spring没法用这个表达式来实现月末功能。

解决方案

解决方法有这几种:

1、还是用Spring的@Scheduled注解,但是改成每天都执行一次,然后使用Hutool的日期工具判断今天是否是当月的最后一天,不然不是直接return即可。

直接上代码:

    @Scheduled(cron = "0 0 8 * * ?")
    public void common() 
        DateTime nowDate = DateUtil.date();
        int nowDay = nowDate.getField(DateField.DAY_OF_MONTH);
        DateTime lastDayOfMonthDateTime = DateUtil.endOfMonth(new Date());
        int lastDayOfMonth = lastDayOfMonthDateTime.getField(DateField.DAY_OF_MONTH);
        if (nowDay != lastDayOfMonth) 
            return;
        
        log.info("月末的早上8点给您统计数据啦!");
    

即每天早上8点都执行,如果当天日期不等于月末日志,直接return即可。这样做对原来的代码改动最小,稍微浪费一点性能。其中的DateUtil来自Hutool,这个方法真的太好用啦!!

2、换Quartz来执行这个定时任务

可以参考SpringBoot整合Quartz_桐花思雨的博客-CSDN博客_springboot整合quartz

来通过Quartz的方式执行定时任务

3、请求产品经理修改一下需求

当然这个只能算不算办法的办法了,就是推迟一下,将月末改为下一个月的第一天去执行,这样就可以避免月末这个问题了,cron表达式就可以改为:0 0 8 1 * ?

相当于每月的1号早上8点去统计上个月的数据。

关于分布式

如果采用1、3中方法还有一个问题,那就是如果采用分布式部署,部署几台就会有几台一起去执行统计任务,很明显会浪费性能甚至导致数据错乱,那如果我只要一台去执行呢?

我这里采用的方法是通过Redis来解决,使用incr一个key的方法,哪台服务器抢到incr后的方法等于1才能执行这个定时任务,其他服务器incr得到的值必定大于1,那么它们就不会去执行了。

以上是关于原创SpringBoot实现月末功能的主要内容,如果未能解决你的问题,请参考以下文章

2019年9月末周java面试总结

MYSQL按日,周和月分组,带有php时间戳[重复]

SQL 分组按周和月在同一时间 (Redshift)

需要在包订阅的日期中添加周、日和月

SQL Server 2016 到 SSMS:固定周结束日期和月结束日期快照日期

我已经使用了 3 个按钮,一个用于一周,一个用于一个月,一个用于一天,如果单击一周按钮,应该禁用日和月