1.Spring Task
1.介绍
Spring Task 是Spring框架提供的任务调度工具,可以按照约定的时间自动执行某个代码逻辑。
作用:定时自动执行某段Java代码
应用场景:
•信用卡每月还款提醒
•银行贷款每月还款提醒
•火车票售票系统处理未支付订单
•入职纪念日为用户发送通知
2.cron表达式
cron表达式其实就是一个字符串,通过cron表达式可以定义任务触发的时间构成规则:分为6或7个域,由空格分隔开,每个域代表一个含义每个域的含义分别为:秒、分钟、小时、日、月、周、年(可选)
日和周(周几),往往只能定义一个,另一个用?
cron表达式在线生成器:在线Cron表达式生成器
3.入门案例
Spring Task使用步骤:
①导入maven坐标 spring-context(已存在)
②启动类添加注解 @EnableScheduling 开启任务调度
③自定义定时任务类
2.订单状态定时处理
/**
* 定时任务类,定时处理订单状态
*/
@Component
@Slf4j
public class OrderTask {
@Autowired
private OrderMapper orderMapper;
/**
* 处理超时订单的方法
*/
@Scheduled(cron = "0 * * * * ? ")//每分钟触发一次
public void processTimeoutOrder(){
log.info("定时处理超时订单:{}", LocalDateTime.now());
LocalDateTime time = LocalDateTime.now().plusMinutes(-15);
//select * from orders where status=? and order_time <(当前时间-15分钟)
List<Orders> ordersList = orderMapper.getByStatusAndOrderTimeLT(Orders.PENDING_PAYMENT, time);
if (ordersList !=null && ordersList.size()>0){
for (Orders orders : ordersList) {
orders.setStatus(Orders.CANCELLED);
orders.setCancelReason("订单超时,自动取消");
orders.setCancelTime(LocalDateTime.now());
orderMapper.update(orders);
}
}
}
/**
* 处理一直处于派送中的订单
*/
@Scheduled(cron = "0 0 1 * * ?")//每天凌晨一点触发一次
public void processDeliveryOrder(){
log.info("定时处理处于派送中的订单:{}",LocalDateTime.now());
LocalDateTime time = LocalDateTime.now().plusMinutes(-60);
//处理上一个工作日一直处于派送中的订单
List<Orders> ordersList = orderMapper.getByStatusAndOrderTimeLT(Orders.DELIVERY_IN_PROGRESS, time);
if (ordersList !=null && ordersList.size()>0){
for (Orders orders : ordersList) {
orders.setStatus(Orders.COMPLETED);
orderMapper.update(orders);
}
}
}
}
3.WebSocket
1.介绍
WebSocket 是基于 TCP 的一种新的网络协议。它实现了浏览器与服务器全双工通信——浏览器和服务器只需要完成一次握手,两者之间就可以创建持久性的连接, 并进行双向数据传输。
HTTP协议和WebSocket协议对比:
•HTTP是短连接
•WebSocket是长连接
•HTTP通信是单向的,基于请求响应模式
•WebSocket支持双向通信
•HTTP和WebSocket底层都是TCP连接
应用场景:视频弹幕、网页聊天、体育实况更新、股票基金报价实时更新
2.入门案例
实现步骤:
①直接使用websocket.html页面作为WebSocket客户端
②导入WebSocket的maven坐标
③导入WebSocket服务端组件WebSocketServer,用于和客户端通信
④导入配置类WebSocketConfiguration,注册WebSocket的服务端组件
⑤导入定时任务类WebSocketTask,定时向客户端推送数据
既然WebSocket支持双向通信,功能看似比HTTP强大,那么我们是不是可以基于WebSocket开发所有的业务功能?
WebSocket缺点:
•服务器长期维护长连接需要一定的成本
•各个浏览器支持程度不一
•WebSocket 是长连接,受网络限制比较大,需要处理好重连
结论:WebSocket并不能完全取代HTTP,它只适合在特定的场景下使用
4.来单提醒
1.需求分析与设计
设计:
•通过WebSocket实现管理端页面和服务端保持长连接状态
•当客户支付后,调用WebSocket的相关API实现服务端向客户端推送消息
•客户端浏览器解析服务端推送的消息,判断是来单提醒还是客户催单,进行相应的消息提示和语音播报
•约定服务端发送给客户端浏览器的数据格式为JSON,字段包括:type,orderId,content
- type 为消息类型,1为来单提醒 2为客户催单
- orderId 为订单id
- content 为消息内容
2.代码开发
OrderServiceImp:
在OrderServiceImpl中注入WebSocketServer对象,修改paySuccess方法,加入如下代码:
5.客户催单
1.需求分析与设计
设计:
•通过WebSocket实现管理端页面和服务端保持长连接状态
•当用户点击催单按钮后,调用WebSocket的相关API实现服务端向客户端推送消息
•客户端浏览器解析服务端推送的消息,判断是来单提醒还是客户催单,进行相应的消息提示和语音播报
•约定服务端发送给客户端浏览器的数据格式为JSON,字段包括:type,orderId,content
- type 为消息类型,1为来单提醒 2为客户催单
- orderId 为订单id
- content 为消息内容
2.代码开发
OrderController
/**
* 用户催单
* @param id
* @return
*/
@GetMapping("/reminder/{id}")
@ApiOperation("用户催单")
public Result reminder(@PathVariable("id") Long id){
orderService.reminder(id);
return Result.success();
}
OrderServiceImpl
/**
* 用户催单
* @param id
*/
public void reminder(Long id) {
//查询订单是否存在
Orders orders = orderMapper.getById(id);
if (orders==null){
throw new OrderBusinessException(MessageConstant.ORDER_STATUS_ERROR);
}
//基于webSocket实现催单
Map map=new HashMap();
map.put("type",2);//2代表用户催单
map.put("orderId",id);
map.put("content","订单号"+orders.getNumber());
webSocketServer.sendToAllClient(JSON.toJSONString(map));
}