大家好,我是Tom哥

今天跟大家聊下定时任务,很多人都喜欢网购,选择商品、下单、付款,然后在家坐等拆包裹,很少有人主动去点 确认收货,那岂不是结束不了订单,卖家也收不到货款。

其实,平台早已想到这个问题,所以会有定时任务,我们只需要设定好 目标执行时间,到了时间后,系统会自动执行 确认收货

接下来,我们来看下,常见的定时任务有哪些?

单点定时任务

1、JDK原生

自从JDK1.5之后,提供了ScheduledExecutorService代替TimerTask来执行定时任务,提供了不错的可靠性。

public class SomeScheduledExecutorService {public static void main(String[] args) {// 创建任务队列,共 10 个线程ScheduledExecutorService scheduledExecutorService =Executors.newScheduledThreadPool(10);// 执行任务: 1秒 后开始执行,每 30秒 执行一次scheduledExecutorService.scheduleAtFixedRate(() -> {System.out.println("执行任务:" + new Date());}, 10, 30, TimeUnit.SECONDS);}
}

2、Spring Task

Spring Framework自带定时任务,提供了cron表达式来实现丰富定时任务配置。

新手推荐使用https://cron.qqe2.com/这个网站来匹配你的cron表达式

@Configuration
@EnableScheduling
public class SomeJob {private static final Logger LOGGER = LoggerFactory.getLogger(SomeJob.class);/*** 每分钟执行一次(例:18:01:00,18:02:00)* 秒 分钟 小时 日 月 星期 年*/@Scheduled(cron = "0 0/1 * * * ? *")public void someTask() {//...}
}

单点的定时服务在目前微服务的大环境下,应用场景越来越局限,所以尝鲜一下分布式定时任务吧。

3、基于 Redis 实现

相较于之前两种方式,这种基于Redis的实现可以通过多点来增加定时任务,多点消费。但是要做好防范重复消费的准备。

3.1 通过ZSet的方式

将定时任务存放到ZSet集合中,并且将过期时间存储到ZSet的Score字段中,然后通过一个循环来判断当前时间内是否有需要执行的定时任务,如果有则进行执行。

具体实现代码如下:

@Configuration
@EnableScheduling
public class RedisJob {public static final String JOB_KEY = "redis.job.task";private static final Logger LOGGER = LoggerFactory.getLogger(RedisJob.class);@Autowired private StringRedisTemplate stringRedisTemplate;/*** 添加任务.** @param task*/public void addTask(String task, Instant instant) {stringRedisTemplate.opsForZSet().add(JOB_KEY, task, instant.getEpochSecond());}/*** 定时任务队列消费* 每分钟消费一次(可以缩短间隔到1s)*/@Scheduled(cron = "0 0/1 * * * ? *")public void doDelayQueue() {long nowSecond = Instant.now().getEpochSecond();// 查询当前时间的所有任务Set<String> strings = stringRedisTemplate.opsForZSet().range(JOB_KEY, 0, nowSecond);for (String task : strings) {// 开始消费 taskLOGGER.info("执行任务:{}", task);}// 删除已经执行的任务stringRedisTemplate.opsForZSet().remove(JOB_KEY, 0, nowSecond);}
}

适用场景如下:

  • 订单下单之后15分钟后,用户如果没有付钱,系统需要自动取消订单。

  • 红包24小时未被查收,需要延迟执退还业务;

  • 某个活动指定在某个时间内生效&失效;

优势是:

  • 省去了MySQL的查询操作,而使用性能更高的Redis做为代替;

  • 不会因为停机等原因,遗漏要执行的任务;

3.2 键空间通知的方式

我们可以通过Redis的键空间通知来实现定时任务,它的实现思路是给所有的定时任务设置一个过期时间,等到了过期之后,我们通过订阅过期消息就能感知到定时任务需要被执行了,此时我们执行定时任务即可。

默认情况下Redis是不开启键空间通知的,需要我们通过config set notify-keyspace-events Ex的命令手动开启。开启之后定时任务的代码如下:

自定义监听器

public class KeyExpiredListener extends KeyExpirationEventMessageListener {public KeyExpiredListener(RedisMessageListenerContainer listenerContainer) {super(listenerContainer);}@Overridepublic void onMessage(Message message, byte[] pattern) {// channelString channel = new String(message.getChannel(), StandardCharsets.UTF_8);// 过期的keyString key = new String(message.getBody(), StandardCharsets.UTF_8);// todo 你的处理}
}

设置该监听器

@Configuration
public class RedisExJob {@Autowired private RedisConnectionFactory redisConnectionFactory;@Beanpublic RedisMessageListenerContainer redisMessageListenerContainer() {RedisMessageListenerContainer redisMessageListenerContainer = new RedisMessageListenerContainer();redisMessageListenerContainer.setConnectionFactory(redisConnectionFactory);return redisMessageListenerContainer;}@Beanpublic KeyExpiredListener keyExpiredListener() {return new KeyExpiredListener(this.redisMessageListenerContainer());}
}

Spring会监听符合以下格式的Redis消息

private static final Topic TOPIC_ALL_KEYEVENTS = new PatternTopic("__keyevent@*");

基于Redis的定时任务能够适用的场景也比较有限,但实现上相对简单,但对于功能幂等有很大要求。从使用场景上来说,更应该叫做延时任务。

场景举例:

  • 订单下单之后15分钟后,用户如果没有付钱,系统需要自动取消订单。

  • 红包24小时未被查收,需要延迟执退还业务;

优劣势是:

  • 被动触发,对于服务的资源消耗更小;

  • Redis的Pub/Sub不可靠,没有ACK机制等,但是一般情况可以容忍;

  • 键空间通知功能会耗费一些CPU

分布式定时任务

引入分布式定时任务组件or中间件

将定时任务作为单独的服务,遏制了重复消费,独立的服务也有利于扩展和维护。

1、quartz

依赖于MySQL,使用相对简单,可多节点部署,通过竞争数据库锁来保证只有一个节点执行任务。没有图形化管理页面,使用相对麻烦。

2、elastic-job-lite

依赖于Zookeeper,通过zookeeper的注册与发现,可以动态的添加服务器。

  • 多种作业模式

  • 失效转移

  • 运行状态收集

  • 多线程处理数据

  • 幂等性

  • 容错处理

  • 支持spring命名空间

  • 有图形化管理页面

3、LTS

依赖于Zookeeper,集群部署,可以动态的添加服务器。可以手动增加定时任务,启动和暂停任务。

  • 业务日志记录器

  • SPI扩展支持

  • 故障转移

  • 节点监控

  • 多样化任务执行结果支持

  • FailStore容错

  • 动态扩容

  • 对spring相对友好

  • 有监控和管理图形化界面

4、xxl-job

国产,依赖于MySQL,基于竞争数据库锁保证只有一个节点执行任务,支持水平扩容。可以手动增加定时任务,启动和暂停任务。

  • 弹性扩容

  • 分片广播

  • 故障转移

  • Rolling实时日志

  • GLUE(支持在线编辑代码,免发布)

  • 任务进度监控

  • 任务依赖

  • 数据加密

  • 邮件报警

  • 运行报表

  • 优雅停机

  • 国际化(中文友好)

总结

微服务下,推荐使用xxl-job这一类组件服务将定时任务合理有效的管理起来。而单点的定时任务有其局限性,适用于规模较小、对未来扩展要求不高的服务。

相对而言,基于spring task的定时任务最简单快捷,而xxl-job的难度主要体现在集成和调试上。无论是什么样的定时任务,你都需要确保:

  • 任务不会因为集群部署而被多次执行。

  • 任务发生异常得到有效的处理

  • 任务的处理过慢导致大量积压

  • 任务应该在预期的时间点执行

中间件可以将服务解耦,但增加了复杂度


关于我:

Tom哥,计算机研究生,校招进阿里,期间还拿了百度、华为、中兴、腾讯 等6家大厂offer,P7 技术专家。出过专利,CSDN博客专家。

多年的大厂浸染,参加多次淘宝双11大促活动,在系统架构方面有丰富经验,沉淀总结在公众号:微观技术

他专注于微服务、高并发、高性能缓存、分布式架构、高可用、团队管理等方面,喜欢挖掘开源框架亮点设计,内容都是面试官喜欢考察的。强烈推荐关注一波。

淘宝超时确认收货 是 如何实现?相关推荐

  1. 天猫,淘宝,京东收货信息中,自动识别手机号、姓名、省市区

    最近工作需要,需要从一串字符串中识别出姓名,手机号和省市区,例如下面的一些字符串: 橱之友,包卫贞,13600000000,浙江省宁波市,慈溪市,庵东镇 杭州湾新区世纪城翠湖苑5栋2000 [识别出] ...

  2. 从零开发短视频电商 30分钟未支付订单自动关闭、七天自动确认收货等延迟任务问题

    文章目录 常见延迟任务 常见解决方案 主动形式 被动形式 基于Redis实现ZSet的方式.键空间通知的方式 ZSet的方式 键空间通知的方式 RocketMQ延迟消息 延迟消息级别配置 客户端发送延 ...

  3. 未来5年,淘宝无货源带货是取代房子最好的副业!

    近来一直有这样一个说法:"下一个暴利风口来了,将造就大批富翁,抓住就有机会致富!" 看看朋友圈那些有钱人,几乎都是因为踩对了风口,赚得盆满钵满: 15年前,咬咬牙供几套房,或许一辈 ...

  4. mysql怎么实现确认收货_Tp结合redis实现订单自动收货

    做的电商平台,用户在收到货之后,大部分都不会主动的点击确认收货,导致给商家结款的时候,商家各种投诉,于是就根据需求,要做一个订单在发货之后的x天自动确认收货.所谓的订单自动确认收货,就是在在特定的时间 ...

  5. 闲鱼无法确认收货显示服务器繁忙,闲鱼不确认收货怎么办?解决办法都是这样的...

    现在网购这件事已经是一种很普遍的生活方式了,越来越多的人加入了这种便捷的生活方式.但是买多了会有人发现,会出现东西还是很不错但自己不怎么使用或者是干脆在家占地方的一些物品,所以有很多人将这类型的物品放 ...

  6. 艾司博讯:拼多多没确认收货多久系统才自动确认收货

    现在大多朋友对网购都不陌生,一般我们收到货后都会确认收货,然后给商品进行评价.当然,一些朋友比较忙或者没有确认收货的习惯,过了一段时间后会自动确认收货.那么拼多多确认收货期限是几天,下面就为大家带来介 ...

  7. 03确认收货与自动收货

    3.确认收货与自动收货 3.1自动收货 3.1.1需求分析与实现思路 当物流公司将货物送到了用户收货地址之后,需要用户点击确认收货,当用户点击了确认收货之后,会修改订单状态为已完成 3.1.2 代码实 ...

  8. 福维克扫地机器人清扫不干净_福维克扫地机器人VR200怎么样?确认收货三个月,诉说一下使用过程!...

    福维克扫地机器人VR200怎么样?确认收货三个月,诉说一下使用过程!:了解福维克扫地机器人VR200怎么样?我是在小红书看中的福维克扫地机器人VR200,看过百度知道很多关于福维克扫地机器人VR200 ...

  9. php 10天自动确认收货,ECSHOP修正recieve.php判断不严谨导致邮件自动确认收货

    今天小编在一个技术论坛上看了这样一个求助帖,一位用ecshop建站的店长朋友反应"用户明明没有确认收货,可是会员管理里面,订单状态却显示为已经确认收货".看到这个求助帖之后,小编很 ...

  10. php 10天自动确认收货,修正Ecshop的recieve.php邮件自动确认收货BUG

    Ecshop没有点击邮件中的确认收货链接,但是进入个人中心却发现订单状态已经更改为了已经收到货的状态了.问题出在QQ邮箱.Ecshop的recieve设计是用户点了链接直接就改变订单状态到确认收货,而 ...

最新文章

  1. Eclipse+Maven+SpringMVC+Mybatis+MySql搭建总结
  2. 凡客诚品-工作经历 程序员你有一个感恩的心吗?
  3. 关于linux分区与挂载的解释
  4. HBase BlockCache系列 - 探求BlockCache实现机制
  5. 99%的程序员都在用Lombok,原理竟然这么简单?我也手撸了一个!|建议收藏!!!...
  6. SQL Server 2008R2安装详细教程(附安装包)
  7. Spring 核心控制器DispatcherServlet(二)
  8. CFS调度主要代码分析一
  9. srt字幕转ass字幕在线工具分享
  10. 使用OpenRefine清洗数据实例
  11. 最新抖音视频无水印解析接口及原理
  12. Node.js常用console中的几种方法
  13. ProxySQL 入门教程
  14. css的定位属性——固定定位和粘性定位
  15. xshell复制粘贴快捷键
  16. 集合的基本运算及文氏图
  17. matlab nan变成0_最优化计算与matlab实现(17)——粒子群优化算法——带压缩因子的粒子群算法...
  18. 看了我的mybatis-plus用法,同事也开始悄悄模仿了
  19. 小写数字转换成大写数字
  20. mysql frm ibd 创建表_MySQL数据库实现从.frm文件和.ibd文件恢复数据表方法

热门文章

  1. SpringBoot之整合thymeleaf渲染Web页面
  2. UOJ #449.【集训队作业2018】喂鸽子 min-max容斥
  3. Oracle表空间增加方法
  4. Transition 总结
  5. 二级c语言笔试试题及答案,1996年9月全国计算机等级考试二级C语言笔试试题及答案...
  6. DXF读写:标注样式组码中文说明
  7. 数据挖掘研究的机遇及挑战 洪胜宏
  8. 美团的2020年:千亿美元帝国的贪吃蛇游戏,气势汹汹也危机重重
  9. 软件开发入门【3分钟课程】
  10. UIKit框架的介绍