SSM开发书评网27:Spring-Task定时任务,来实现【更新评分、评价人数】的功能;(其中,包括Cron表达式,@Scheduled任务调度注解)

本文主要是介绍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任务调度注解)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/452668

相关文章

Spring Security基于数据库的ABAC属性权限模型实战开发教程

《SpringSecurity基于数据库的ABAC属性权限模型实战开发教程》:本文主要介绍SpringSecurity基于数据库的ABAC属性权限模型实战开发教程,本文给大家介绍的非常详细,对大... 目录1. 前言2. 权限决策依据RBACABAC综合对比3. 数据库表结构说明4. 实战开始5. MyBA

AJAX请求上传下载进度监控实现方式

《AJAX请求上传下载进度监控实现方式》在日常Web开发中,AJAX(AsynchronousJavaScriptandXML)被广泛用于异步请求数据,而无需刷新整个页面,:本文主要介绍AJAX请... 目录1. 前言2. 基于XMLHttpRequest的进度监控2.1 基础版文件上传监控2.2 增强版多

Spring Security方法级安全控制@PreAuthorize注解的灵活运用小结

《SpringSecurity方法级安全控制@PreAuthorize注解的灵活运用小结》本文将带着大家讲解@PreAuthorize注解的核心原理、SpEL表达式机制,并通过的示例代码演示如... 目录1. 前言2. @PreAuthorize 注解简介3. @PreAuthorize 核心原理解析拦截与

一文详解JavaScript中的fetch方法

《一文详解JavaScript中的fetch方法》fetch函数是一个用于在JavaScript中执行HTTP请求的现代API,它提供了一种更简洁、更强大的方式来处理网络请求,:本文主要介绍Jav... 目录前言什么是 fetch 方法基本语法简单的 GET 请求示例代码解释发送 POST 请求示例代码解释

Java图片压缩三种高效压缩方案详细解析

《Java图片压缩三种高效压缩方案详细解析》图片压缩通常涉及减少图片的尺寸缩放、调整图片的质量(针对JPEG、PNG等)、使用特定的算法来减少图片的数据量等,:本文主要介绍Java图片压缩三种高效... 目录一、基于OpenCV的智能尺寸压缩技术亮点:适用场景:二、JPEG质量参数压缩关键技术:压缩效果对比

Java调用C++动态库超详细步骤讲解(附源码)

《Java调用C++动态库超详细步骤讲解(附源码)》C语言因其高效和接近硬件的特性,时常会被用在性能要求较高或者需要直接操作硬件的场合,:本文主要介绍Java调用C++动态库的相关资料,文中通过代... 目录一、直接调用C++库第一步:动态库生成(vs2017+qt5.12.10)第二步:Java调用C++

Redis分片集群的实现

《Redis分片集群的实现》Redis分片集群是一种将Redis数据库分散到多个节点上的方式,以提供更高的性能和可伸缩性,本文主要介绍了Redis分片集群的实现,具有一定的参考价值,感兴趣的可以了解一... 目录1. Redis Cluster的核心概念哈希槽(Hash Slots)主从复制与故障转移2.

springboot+dubbo实现时间轮算法

《springboot+dubbo实现时间轮算法》时间轮是一种高效利用线程资源进行批量化调度的算法,本文主要介绍了springboot+dubbo实现时间轮算法,文中通过示例代码介绍的非常详细,对大家... 目录前言一、参数说明二、具体实现1、HashedwheelTimer2、createWheel3、n

使用Python实现一键隐藏屏幕并锁定输入

《使用Python实现一键隐藏屏幕并锁定输入》本文主要介绍了使用Python编写一个一键隐藏屏幕并锁定输入的黑科技程序,能够在指定热键触发后立即遮挡屏幕,并禁止一切键盘鼠标输入,这样就再也不用担心自己... 目录1. 概述2. 功能亮点3.代码实现4.使用方法5. 展示效果6. 代码优化与拓展7. 总结1.

Mybatis 传参与排序模糊查询功能实现

《Mybatis传参与排序模糊查询功能实现》:本文主要介绍Mybatis传参与排序模糊查询功能实现,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、#{ }和${ }传参的区别二、排序三、like查询四、数据库连接池五、mysql 开发企业规范一、#{ }和${ }传参的