Java微服务下的分布式事务介绍及其解决方案2
内容导读
互联网集市收集整理的这篇技术教程文章主要介绍了Java微服务下的分布式事务介绍及其解决方案2,小编现在分享给大家,供广大互联网技能从业者学习和参考。文章包含6071字,纯文字阅读大概需要9分钟。
内容图文
1.前言
本文将详细介绍分布式的解决方案–消息队列实现分布式事务的解决方案
2.业务场景介绍
我们模拟慕课网付费课程的下单,你在慕课网买了视频后,你的学习列表要订单服务要更新你支付的状态,此外,学习服务要有你的添加选课信息
3.解决方案
下面我详细介绍下,这个解决方案
1、支付成功后,订单服务向本地数据库更新订单状态并向消息表写入“添加选课消息”,通过本地数据库保证订单状态和添加选课消息的事务。。
2、定时任务扫描消息表,取出“添加选课任务“并发向MQ。
3、学习服务接收到添加选课的消息,先查询本地数据库的历史消息表是否存在消息,存在则说明已经添加选课,否则向本地数据库添加选课,并向历史消息表添加选课消息。这里选课表和历史消息表在同一个数据库,通过本地事务保证。
4、学习服务接收到添加选课的消息,通过查询消息表判断如果已经添加选课也向MQ发送“完成添加选课任务的消息”,否则则添加选课,完成后向MQ发送“完成添加选课任务的消息”,
5、订单服务接收到完成选课的消息后删除订单数据库中消息表的“添加选课消息”,为保证后期对账将消息表的消息先添加到历史消息表再删除消息,表示此消息已经完成。
需要用到的几个表
1.待处理的任务表,在订单系统里面保存在这里插入图片描述
2.已经完成的任务表:订单和学习系统都要有
3.选课信息表
4.解决方案的伪代码
4.1订单服务定时发送消息
定时任务发送消息流程如下:
0、每隔1分钟扫描一次任务表。
1、定时任务扫描task表,一次取出多个任务,取出超过1分钟未处理的任务
2、考虑订单服务可能集群部署,为避免重复发送任务使用乐观锁的方式每次从任务列表取出要处理的任务
3、任务发送完毕更新任务发送时间
关于任务表的添加:
正常的流程是订单支付成功向更新订单支付状态并向任务表写入“添加选课任务”。
(没有写用户下单,直接默认有用户下单)
订单服务定时发送消息的代码(用的是SpringTask)
//每隔一分钟扫描消息表,向mq发送消息
@Scheduled(cron="0/3 * * * * *")
public void sendChoosecourseTask(){
//取出当前一分钟之前的时间
Calendar calendar =new GregorianCalendar();
calendar.setTime(new Date());
calendar.add(GregorianCalendar.MINUTE,-1);
Date time = calendar.getTime();
List<XcTask> taskList = taskService.findTaskList(time, 100);
//遍历任务表,发送选课信息
if (taskList!=null && taskList.size()>0){
for (XcTask xcTask : taskList){
//发送选课信息
taskService.publish(xcTask,xcTask.getMqExchange(),xcTask.getMqRoutingkey());
//记录日子
LOGGER.info("send choose course task id:{}",xcTask.getId());
}
}
}
此时已经检测到有人下单支付了,就要发送,取出“添加选课任务“并发向MQ,代码如下:
//发送消息
@Transactional
public void publish(XcTask xcTask,String ex,String routingKey){
//查询任务,如果有任务
Optional<XcTask> optional = xcTaskRepository.findById(xcTask.getId());
if (optional.isPresent()){
//发送消息
rabbitTemplate.convertAndSend(ex,routingKey,xcTask);
//更新时间
XcTask xcTask1 = optional.get();
xcTask1.setUpdateTime(new Date());
xcTaskRepository.save(xcTask1);
}
}
这时候订单系统前期要处理的事(监听用户下单,发送消息到mq)已经完成了,下面就需要学习系统的服务来处理事情了(接收消息,添加选课,发送添加选课成功)
添加选课的代码:
备注:向xc_learning_course添加记录,为保证不重复添加选课,先查询历史任务表,如果从历史任务表查询不到任务说明此任务还没有处理,此时则添加选课并添加历史任务。
//添加选课
@Transactional
public ResponseResult addCourse(String userId, String courseId, String valid, Date startTime, Date endTime, XcTask xcTask){
//校验参数
//查询历史任务,如果历史任务有,说明已经添加过选课了,没有则添加
//查询历史任务
Optional<XcTaskHis> optional = xcTaskHisRepository.findById(xcTask.getId());
if(optional.isPresent()){
return new ResponseResult(CommonCode.SUCCESS);
}
//历史表里面没有,要添加课程
XcLearningCourse xcLearningCourse = xcLearningCourseRepository.findXcLearningCourseByUserIdAndCourseId(userId, courseId);
if (xcLearningCourse == null) {//没有选课记录则添加
xcLearningCourse = new XcLearningCourse();
xcLearningCourse.setUserId(userId);
xcLearningCourse.setCourseId(courseId);
xcLearningCourse.setValid(valid);
xcLearningCourse.setStartTime(startTime);
xcLearningCourse.setEndTime(endTime);
xcLearningCourse.setStatus("501001");
xcLearningCourseRepository.save(xcLearningCourse);
} else {//有选课记录则更新日期
xcLearningCourse.setValid(valid);
xcLearningCourse.setStartTime(startTime);
xcLearningCourse.setEndTime(endTime);
xcLearningCourse.setStatus("501001");
xcLearningCourseRepository.save(xcLearningCourse);
}
//向历史记录表插入记录
Optional<XcTaskHis> optional1= xcTaskHisRepository.findById(xcTask.getId());
if(!optional1.isPresent()){
//添加历史任务
XcTaskHis xcTaskHis = new XcTaskHis();
BeanUtils.copyProperties(xcTask,xcTaskHis);
xcTaskHisRepository.save(xcTaskHis);
}
return new ResponseResult(CommonCode.SUCCESS);
}
接收选课的消息:
接收到添加选课的消息调用添加选课方法完成添加选课,并发送完成选课消息。
@RabbitListener(queues = RabbitMQConfig.XC_LEARNING_ADDCHOOSECOURSE)
public void receiveChoosecourseTask(XcTask xcTask){
//取出消息内容
//取出消息的内容
String requestBody = xcTask.getRequestBody();
Map map = JSON.parseObject(requestBody, Map.class);
String userId = (String) map.get("userId");
String courseId = (String) map.get("courseId");
//添加选课
ResponseResult responseResult = learningService.addCourse(userId, courseId, null, null, null, xcTask);
if (responseResult.isSuccess()){
//添加选课成功,向mq发送完成添加选课的消息
rabbitTemplate.convertAndSend(RabbitMQConfig.EX_LEARNING_ADDCHOOSECOURSE,RabbitMQConfig.XC_LEARNING_FINISHADDCHOOSECOURSE_KEY,xcTask);
}
}
到这时候,学习系统做的事(接收订单系统发过来的消息,添加课程信息,给订单信息发送已经添加课程成功的消息),接下来就要回到订单系统来接收添加课程成功的消息了,将任务从当前任务表删除,将完成的任务添加到完成任务表。
/**
* 完成任务
* @param xcTask
*/
@RabbitListener(queues = RabbitMQConfig.XC_LEARNING_FINISHADDCHOOSECOURSE)
public void receiveFinishChoosecourseTask(XcTask xcTask){
if(xcTask!=null && StringUtils.isNotEmpty(xcTask.getId())){
taskService.finishTask(xcTask.getId());
}
}
//下面是删除任务的代码
//删除任务
@Transactional
public void finishTask(String taskId){
Optional<XcTask> taskOptional = xcTaskRepository.findById(taskId);
if (taskOptional.isPresent()){
XcTask xcTask = taskOptional.get();
xcTask.setDeleteTime(new Date());
XcTaskHis xcTaskHis = new XcTaskHis();
BeanUtils.copyProperties(xcTask,xcTaskHis);
//把记录添加到历史任务里面,为了以前对账能够清楚点,即把任务表的数据复制到历史任务1表里面
xcTaskHisRepository.save(xcTaskHis);
xcTaskRepository.delete(xcTask);
}
}
内容总结
以上是互联网集市为您收集整理的Java微服务下的分布式事务介绍及其解决方案2全部内容,希望文章能够帮你解决Java微服务下的分布式事务介绍及其解决方案2所遇到的程序开发问题。 如果觉得互联网集市技术教程内容还不错,欢迎将互联网集市网站推荐给程序员好友。
内容备注
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 gblab@vip.qq.com 举报,一经查实,本站将立刻删除。
内容手机端
扫描二维码推送至手机访问。