本文主要是介绍SSM开发书评网27:Spring-Task定时任务,来实现【更新评分、评价人数】的功能;(其中,包括Cron表达式,@Scheduled任务调度注解),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
说明:
(1)本篇博客的内容说明:利用Spring-Task模块,实现对所有图书【更新评分、评价人数】的功能;
(2)本篇博客的几个点:
● Mybatis-Plus可以帮助我们生成SQL语句;对于基本的增删改查,Mybatis-Plus是OK的;但是,对于复杂的SQL语句,Mybatis-Plus就比较吃力了;又因为,Mybatis-Plus只是扩展了Mybatis,其并没有修改Mybatis的任何内容;所以,我们在这儿,依旧可以使用Mybatis的内容;自然,这儿需要遵循Mybatis的开发规范,该对应的要对应好;
● 本篇博客的演示了Spring Task的基本使用;这是本篇博客的重点;
● 本篇博客的基本逻辑是:【分析业务逻辑,得到底层SQL】→【根据业务逻辑,编写业务代码】→【创建实现定时任务的类,调用业务逻辑代码,设置定时任务】;
目录
一:【Spring-Task】定时任务模块,简介;
1.定时任务,简介;
2.Spring-Task定时任务模块;
3.Cron表达式;
二:编写【更新评分、评价人数】的逻辑代码;
0.情况说明;
1.【更新评分、评价人数】的:SQL分析;
2.在BookMapper接口中,定义操作底层数据库,实现【更新评分、评价人数】的方法:updateEvaluation()方法;
3.在book.xml中,编写【更新评分、评价人数】的SQL;
4.在BookService接口中,定义【更新评分、评价人数】的方法:updateEvaluation()方法;(PS:Service中的可以起其他名字,无需和Mapper中的保持一致)
5.在BookServiceImpl实现类中,去实现【更新评分、评价人数】的方法:updateEvaluation()方法;
三:【Spring-Task】定时任务模块:在本项目上的应用;
1.不需要额外引入依赖;
2.在applicationContext.xml配置文件中,配置:去开始Spring Task定时任务注解模式;
3.创建task包,创建ComputeTask类:调用Service中的逻辑,设置定时任务;
4.启动Tomcat,观察效果;
一:【Spring-Task】定时任务模块,简介;
1.定时任务,简介;
定时任务概念:
我们想在 [几点几分几秒] 或者 [一个固定的时间间隔内] ,执行Java中的某一段代码;
定时任务是一种非常常见的应用场景:
比如,开发一个闹钟应用,设置其每天早上6:30时,自动播放闹铃,此时就是定时任务;
再比如,金融业中:银行每天需要根据当天的业务情况,生成数据统计报表;又因为,银行每天的业务量可能百万千万,数据量如此庞大,计 算这些业务,生成统计报表,是需要花费一定的时间;所以,我们不能在需要统计报表的时,即时先算;为此,银行就提出了一种延时处理的方案:通常在凌晨2:00-4:00,由银行的服务器自动执行处理数据的任务(这在银行的专业术语中,被称为日终处理),生成统计报表;第二天,当需要查看昨天的业务的统计报表时,只需要直接读取昨晚生成的数据就行了,而不需要现算;那么,如果保证在每天的凌晨2:00-4:00,去执行统计任务呐?其底层就是通过一些定时任务来处理的;
2.Spring-Task定时任务模块;
3.Cron表达式;
说明:
(1)Cron表达式,不是Java独有的技术;
(2)Cron表达式,是一个最多7位的字符串;这7位,分别对应秒、分、小时、日、月、星期、年;
(3)其中,第7位年,可以省略;日和星期是互斥的,即,写了日,星期就只能是?;写了星期,日就只能是?;(如果星期哪儿是?,就表示忽略星期)
(4)如上图中的第一个例子:表示:在任意年的、任意月、任意日、任意小时、任意分、0的时候,执行一次任务;;;;其实,也就是每分钟执行一次任务;
(5)上图的第二个例子:表示:在2000年、任意月、任意日、任意小时、每小时的前五分钟、第0和第30秒,执行任务;(这儿可以看到,我们可以【0,30】这样写,来指定分散的时间点)
(6)上图的第三个例子:表示:在任意年、任意月、星期三、第9-18小时、第0分、第0秒,执行任务;;;;其实,也就是每周三,上午9点到下午6点,整点的时候,执行任务;(这儿可以看到,当我们写了星期,就不能写日了;)
二:编写【更新评分、评价人数】的逻辑代码;
0.情况说明;
在前面,会员可以对图书进行评价;
那么,在新增评分的时候,需要重新计算一下该图书的评分;
……………………………………………………
此时,我们就可以使用【Spring-Task】,来实现;每分钟,来重新计算图书的平均评分;
1.【更新评分、评价人数】的:SQL分析;
这儿的业务主要是:查询某本书的【平均评分】和【评价数量】;下面的SQL,就可以完成这个功能;
UPDATE book b SET evaluation_score =( SELECT IFNULL(AVG(score),0) FROM evaluation WHERE book_id = b.book_id AND state = 'enable' ), evaluation_quantity=( SELECT IFNULL(COUNT(*),0) FROM evaluation WHERE book_id = b.book_id AND state = 'enable' )
说明:
(1)上面的SQL很简单,其中涉及到了AVG,COUNT聚合函数(如有需要,可以参考【数据库的高级查询二:聚合函数(SUM、MAX,MIN,AVG,COUNT)】);子查询(如有需要,可以参考【数据库的高级查询七:子查询】);;;;上面的SQL,很简单,没什么好说的;
(2)但是,上面有一个需要注意的地方:
(3)目前,暂时不要过于考虑数据库优化的内容;
至此,自动计算评分和评价人数的SQL语句就OK了;但是,我们希望,每分钟去执行一次这个SQL;在Spring-Task模块中,如何使用嘞?
2.在BookMapper接口中,定义操作底层数据库,实现【更新评分、评价人数】的方法:updateEvaluation()方法;
package com.imooc.reader.mapper;import com.baomidou.mybatisplus.core.mapper.BaseMapper; import com.imooc.reader.entity.Book;public interface BookMapper extends BaseMapper<Book>{/*** 更新图书评分、评价数量*/public void updateEvaluation(); }
说明:
(1)因为【更新评分、评价人数】的SQL逻辑比较复杂;Mybatis的BaseMapper接口中,没有对应的可以处理这个业务的方法;所以,我们需要自己定义一个方法
3.在book.xml中,编写【更新评分、评价人数】的SQL;
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE mapperPUBLIC "-//mybatis.org//DTD Mapper 3.0//EN""http://mybatis.org/dtd/mybatis-3-mapper.dtd"> <mapper namespace="com.imooc.reader.mapper.BookMapper"><update id="updateEvaluation">UPDATE book b SET evaluation_score =(SELECT IFNULL(AVG(score),0) FROM evaluation WHERE book_id = b.book_id AND state = 'enable'),evaluation_quantity=(SELECT IFNULL(COUNT(*),0) FROM evaluation WHERE book_id = b.book_id AND state = 'enable')</update> </mapper>
说明:
(1)Mybatis-Plus可以帮助我们生成SQL语句;对于基本的增删改查,Mybatis-Plus是OK的;但是,对于上面的那种复杂的SQL语句,Mybatis-Plus就比较吃力了;
(2)又因为,Mybatis-Plus只是扩展了Mybatis,其并没有修改Mybatis的任何内容;所以,我们在这儿,依旧可以使用Mybatis的内容;
(3)自然,这儿需要遵循Mybatis的开发规范,该对应的要对应好:
4.在BookService接口中,定义【更新评分、评价人数】的方法:updateEvaluation()方法;(PS:Service中的可以起其他名字,无需和Mapper中的保持一致)
5.在BookServiceImpl实现类中,去实现【更新评分、评价人数】的方法:updateEvaluation()方法;
【更新评分、评价人数】逻辑代码写完了;那么我们如何每分钟,执行Service层中定义的updateEvaluation()方法呐?
这就需要Spring-Task模块了;
三:【Spring-Task】定时任务模块:在本项目上的应用;
1.不需要额外引入依赖;
2.在applicationContext.xml配置文件中,配置:去开始Spring Task定时任务注解模式;
说明:
(1)聪明如我,在前面我们开发的时候,就引入了task命名空间;
3.创建task包,创建ComputeTask类:调用Service中的逻辑,设置定时任务;
ComputeTask:
package com.imooc.reader.task;import com.imooc.reader.service.BookService; import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Component;import javax.annotation.Resource;@Component public class ComputeTask {@Resourceprivate BookService bookService;//任务调度@Scheduled(cron = "0 * * * * ?")public void updateEvaluation() {bookService.updateEvaluation();System.out.println("已经更新所有图书的评分、评价人数");} }
说明:
(1)因为这个类不是Service,不是Controller;而我们又想让IoC容器管理这个类的对象,所以我们使用了@Component注解;有关@Component这种组件注解,如果有需要,可以参考【Spring IoC容器与Bean管理21:使用注解方式实现Spring IoC二:组件类型注解(对象实例化);@Repository,@Service,@Controller,@Component;】;
(1.2)由(1)联想到,在SSM项目中;Mapper接口没有使用@Repository注解;对于这个问题,可以参考【彻底搞懂使用MyBatis时为什么Dao层不需要@Repository】;这篇文章写的很清楚;
(3)方法说明;
4.启动Tomcat,观察效果;
经过在页面上实操,发现【评分和评价人数】,实现了在每分钟的第0秒实现更新的功能;
而且在控制台这儿,也可以看到每分钟的第0秒,也输出了【已经更新所有图书的评分、评价人数】;
至此,【慕课书评网】的前台都已经完成了;(前台是给用户用的)
接下来,我们要开发【慕课书评网】的后台了;(后台是给管理用的)
这篇关于SSM开发书评网27:Spring-Task定时任务,来实现【更新评分、评价人数】的功能;(其中,包括Cron表达式,@Scheduled任务调度注解)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!