苍穹外卖(十)订单状态定时处理、来单提醒和客户催单

苍穹外卖-day10

课程内容

  • Spring Task
  • 订单状态定时处理
  • WebSocket
  • 来单提醒
  • 客户催单

功能实现:订单状态定时处理来单提醒客户催单

订单状态定时处理:

image-20221218204021760

来单提醒:

image-20221218204119663

客户催单:

image-20221218204202847

1. Spring Task

1.1 介绍

Spring Task 是Spring框架提供的任务调度工具,可以按照约定的时间自动执行某个代码逻辑。

**定位:**定时任务框架

**作用:**定时自动执行某段Java代码

image-20221218183054818 为什么要在Java程序中使用Spring Task?

应用场景:

1). 信用卡每月还款提醒

image-20221218183213088

2). 银行贷款每月还款提醒

image-20221218183410430

3). 火车票售票系统处理未支付订单

image-20221218183614351

4). 入职纪念日为用户发送通知

image-20221218183655186

**强调:**只要是需要定时处理的场景都可以使用Spring Task

1.2 cron表达式

cron表达式其实就是一个字符串,通过cron表达式可以定义任务触发的时间

**构成规则:**分为6或7个域,由空格分隔开,每个域代表一个含义

每个域的含义分别为:秒、分钟、小时、日、月、周、年(可选)

举例:

2022年10月12日上午9点整 对应的cron表达式为:0 0 9 12 10 ? 2022

image-20221218184412491

说明:一般的值不同时设置,其中一个设置,另一个用?表示。

**比如:**描述2月份的最后一天,最后一天具体是几号呢?可能是28号,也有可能是29号,所以就不能写具体数字。

为了描述这些信息,提供一些特殊的字符。这些具体的细节,我们就不用自己去手写,因为这个cron表达式,它其实有在线生成器。

cron表达式在线生成器:https://cron.qqe2.com/

image-20221218184959888

可以直接在这个网站上面,只要根据自己的要求去生成corn表达式即可。所以一般就不用自己去编写这个表达式。

通配符:

* 表示所有值;

? 表示未说明的值,即不关心它为何值;

- 表示一个指定的范围;

, 表示附加一个可能值;

/ 符号前表示开始时间,符号后表示每次递增的值;

cron表达式案例:

*/5 * * * * ? 每隔5秒执行一次

0 */1 * * * ? 每隔1分钟执行一次

0 0 5-15 * * ? 每天5-15点整点触发

0 0/3 * * * ? 每三分钟触发一次

0 0-5 14 * * ? 在每天下午2点到下午2:05期间的每1分钟触发

0 0/5 14 * * ? 在每天下午2点到下午2:55期间的每5分钟触发

0 0/5 14,18 * * ? 在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟触发

0 0/30 9-17 * * ? 朝九晚五工作时间内每半小时

0 0 10,14,16 * * ? 每天上午10点,下午2点,4点

1.3 入门案例

1.3.1 Spring Task使用步骤

1). 导入maven坐标 spring-context(已存在)

image-20221218193251182

2). 启动类添加注解 @EnableScheduling 开启任务调度

3). 自定义定时任务类

1.3.2 代码开发

编写定时任务类:

进入sky-server模块中

package com.sky.task;

import lombok.extern.slf4j.Slf4j;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;

import java.util.Date;

/**
 * 自定义定时任务类
 */
@Component
@Slf4j
public class MyTask {
   
   

    /**
     * 定时任务 每隔5秒触发一次
     */
    @Scheduled(cron = "0/5 * * * * ?")
    public void executeTask(){
   
   
        log.info("定时任务开始执行:{}",new Date());
    }
}

开启任务调度:

启动类添加注解 @EnableScheduling

package com.sky;

import lombok.extern.slf4j.Slf4j;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cache.annotation.EnableCaching;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.transaction.annotation.EnableTransactionManagement;

@SpringBootApplication
@EnableTransactionManagement //开启注解方式的事务管理
@Slf4j
@EnableCaching
@EnableScheduling
public class SkyApplication {
   
   
    public static void main(String[] args) {
   
   
        SpringApplication.run(SkyApplication.class, args);
        log.info("server started");
    }
}
1.3.3 功能测试

启动服务,查看日志

image-20221218194511420

每隔5秒执行一次。

2.订单状态定时处理

2.1 需求分析

用户下单后可能存在的情况:

  • 下单后未支付,订单一直处于**“待支付”**状态
  • 用户收货后管理端未点击完成按钮,订单一直处于**“派送中”**状态

image-20221218194939516 image-20221218194951963

支付超时的订单如何处理? 派送中的订单一直不点击完成如何处理?

对于上面两种情况需要通过定时任务来修改订单状态,具体逻辑为:

  • 通过定时任务每分钟检查一次是否存在支付超时订单(下单后超过15分钟仍未支付则判定为支付超时订单),如果存在则修改订单状态为“已取消”
  • 通过定时任务每天凌晨1点检查一次是否存在“派送中”的订单,如果存在则修改订单状态为“已完成”

2.2 代码开发

1). 自定义定时任务类OrderTask(待完善):

package com.sky.task;

/**
 * 自定义定时任务,实现订单状态定时处理
 */
@Component
@Slf4j
public class OrderTask {
   
   

    @Autowired
    private OrderMapper orderMapper;

    /**
     * 处理支付超时订单
     */
    @Scheduled(cron = "0 * * * * ?")
    public void processTimeoutOrder(){
   
   
        log.info("处理支付超时订单:{}", new Date());
    }

    /**
     * 处理“派送中”状态的订单
     */
    @Scheduled(cron = "0 0 1 * * ?")
    public void processDeliveryOrder(){
   
   
        log.info("处理派送中订单:{}", new Date());
    }

}

2). 在OrderMapper接口中扩展方法:

	/**
     * 根据状态和下单时间查询订单
     * @param status
     * @param orderTime
     */
    @Select("select * from orders where status = #{status} and order_time < #{orderTime}")
    List<Orders> getByStatusAndOrdertimeLT(Integer status, 
### 苍穹外卖提醒的配置方法 #### 配置概述 苍穹外卖中的来提醒功能通常涉及定时任务、消息推送以及用户交互逻辑的设计。通过分析 sky-server 模块的功能结构,可以发现其主要由多个子模块协同完成业务需求[^1]。 #### 定时任务实现 在 `sky-server` 的服务架构中,可以通过 Spring Boot 提供的任务调度机制(@Scheduled 注解)或者 Quartz 调度框架实现来提醒定时触发功能。具体来说,在 `service` 包下的某个 Service 类中定义定时任务的方法,并标注 @Scheduled 注解指定执行频率或时间间隔[^2]。 ```java @Service public class OrderReminderService { @Autowired private OrderMapper orderMapper; @Scheduled(cron = "0/30 * * * * ?") // 每30秒执行一次 public void remindOrders() { List<Order> pendingOrders = orderMapper.findPendingOrders(); for (Order order : pendingOrders) { notifyUser(order.getUserId(), "您有新的订单,请及时处理!"); } } private void notifyUser(Long userId, String message) { // 实现通知逻辑,比如发送短信、邮件或APP内消息 } } ``` 上述代码片段展示了如何利用 Spring 的 @Scheduled 注解创建一个每 30 秒运行一次的任务,用于查询未处理订单并向对应用户发出提醒。 #### 用户与消息推送 对于用户发起的请求,可以在 `controller` 层设计相应的 API 接口接收前端传来的参数;随后调用 `service` 中的具体业务逻辑更新数据库状态并再次触发改用户的提醒流程。 ```java @RestController @RequestMapping("/orders") public class OrderController { @Autowired private OrderService orderService; @PostMapping("/{orderId}/remind") public ResponseEntity<String> remindOrder(@PathVariable Long orderId) { try { orderService.remindOrder(orderId); return new ResponseEntity<>("成功", HttpStatus.OK); } catch (Exception e) { return new ResponseEntity<>(e.getMessage(), HttpStatus.INTERNAL_SERVER_ERROR); } } } ``` 此段代码实现了当商户收到客户促后重新激活提醒的通知链路。 #### 使用内网穿透工具支持远程访问 如果测试环境部署于本地服务器上,则可能需要用到像 cpolar 这样的内网穿透工具以便外部网络能够正常访问应用接口从而模拟真实场景下不同角色之间的互动过程。按照官方文档指引操作即可快速搭建起可用连接通道。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值