RabbitMQ 延迟队列实现订单支付结果异步阶梯性通知
在第三方支付中,例如支付宝、或者微信,对于订单请求,第三方支付系统采用的是消息同步返回、异步通知+主动补偿查询的补偿机制。
例如一个支付结果的通知,一方面会在支付页面跳转时候返回支付结果(一般只用作前端展示使用,非最终状态),同时会采用后台异步通知机制(有前台、后台通知的,以后台异步通知结果为准),但由于前台跳转、后台结果通知都可能失效,因此还以定时补单+请求方主动查询接口作为辅助手段。
常见的补单操作,任务调度策略一般设定30秒、60秒、3分钟、6分钟、10分钟调度多次(以自己业务需要),如果调度接收到响应确认报文,补单成功,则中止对应订单的调度任务;如果超过补单上限次数,则停止补单,避免无谓的资源浪费。请求端随时可以发起请求报文查询对应订单的状态。
所以最终打算使用rabbitmq的消息延迟+死信队列来实现。消息模型如下:
producer发布消息,通过exchangeA的消息会被分发到QueueA,Consumer监听queueA,一旦有消息到来就被消费,这边的消费业务就是通知前端,如果通知失败,就创建一个延迟队列declareQueue,设置每个消息的ttl然后通过declare_exchange将消息分发到declare_queue,因为declare_queue没有consumer并且declare_queue中的消息设置了ttl,当ttl到期后,将通过DEX路由到queueA,被重新消费。
-
package org.delayQueue;
-
-
import com.rabbitmq.client.BuiltinExchangeType;
-
import com.rabbitmq.client.Channel;
-
import com.rabbitmq.client.Connection;
-
import com.rabbitmq.client.ConnectionFactory;
-
-
public class DeclareQueue {
-
public static String EXCHANGE_NAME = "notifyExchange";
-
-
public static void init() {
-
ConnectionFactory factory = new ConnectionFactory();
-
factory.setHost("localhost");
-
factory.setPort(5672);
-
-
Connection connection = null;
-
try {
-
connection = factory.newConnection();
-
Channel channel = connection.createChannel();
-
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.TOPIC);
-
String routingKey = "AliPaynotify";
-
String message = "http://localhost:8080/BossCenter/payGateway/notifyRecv.jsp?is_success=T¬ify_id=4ab9bed148d043d0bf75460706f7774a¬ify_time=2014-08-29+16%3A22%3A02¬ify_type=trade_status_sync&out_trade_no=1421712120109862&total_fee=424.42&trade_no=14217121201098611&trade_status=TRADE_SUCCESS";
-
channel.basicPublish(EXCHANGE_NAME, routingKey, null, message.getBytes());
-
System.out.println(" [x] Sent :" + message);
-
} catch (Exception e) {
-
// TODO Auto-generated catch block
-
e.printStackTrace();
-
} finally {
-
if (connection != null) {
-
try {
-
connection.close();
-
} catch (Exception ignore) {
-
}
-
}
-
}
-
}
-
-
public static void main(String args[]) {
-
init();
-
}
-
-
}
DeclareConsumer.java
-
package org.delayQueue;
-
-
import java.io.BufferedReader;
-
import java.io.IOException;
-
import java.io.InputStreamReader;
-
import java.util.ArrayList;
-
import java.util.HashMap;
-
import java.util.List;
-
import java.util.Map;
-
import java.util.Map.Entry;
-
-
import org.apache.http.HttpResponse;
-
import org.apache.http.client.ClientProtocolException;
-
import org.apache.http.client.HttpClient;
-
import org.apache.http.client.methods.HttpPost;
-
import org.apache.http.impl.client.DefaultHttpClient;
-
-
import com.rabbitmq.client.AMQP;
-
import com.rabbitmq.client.Channel;
-
import com.rabbitmq.client.Connection;
-
import com.rabbitmq.client.ConnectionFactory;
-
import com.rabbitmq.client.Consumer;
-
import com.rabbitmq.client.DefaultConsumer;
-
import com.rabbitmq.client.Envelope;
-
-
public class DeclareConsumer {
-
public static String EXCHANGE_NAME = "notifyExchange";
-
public static String QU_declare_15S = "Qu_declare_15s";
-
public static String EX_declare_15S = "EX_declare_15s";
-
public static String ROUTINGKEY = "AliPaynotify";
-
public static Connection connection = null;
-
public static Channel channel = null;
-
public static Channel DECLARE_15S_CHANNEL = null;
-
public static String declare_queue = "init";
-
public static String originalExpiration = "0";
-
public static void init() throws Exception {
-
ConnectionFactory factory = new ConnectionFactory();
-
factory.setHost("localhost");
-
factory.setPort(5672);
-
connection = factory.newConnection();
-
channel = connection.createChannel();
-
DECLARE_15S_CHANNEL = connection.createChannel();
-
}
-
-
public static void consume() {
-
try {
-
channel.exchangeDeclare(EXCHANGE_NAME, "topic");
-
final String queueName = channel.queueDeclare().getQueue();
-
-
channel.queueBind(queueName, EXCHANGE_NAME, ROUTINGKEY);
-
System.out.println(" [*] Waiting for messages. To exit press CTRL+C");
-
final Consumer consumer = new DefaultConsumer(channel) {
-
@Override
-
public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
-
String message = new String(body, "UTF-8");
-
Map<String, Object> headers = properties.getHeaders();
-
if (headers != null) {
-
List<Map<String, Object>> xDeath = (List<Map<String, Object>>) headers.get("x-death");
-
System.out.println("xDeath--- > " + xDeath);
-
if (xDeath != null && !xDeath.isEmpty()) {
-
Map<String, Object> entrys = xDeath.get(0);
-
// for(Entry<String, Object>
-
// entry:entrys.entrySet()){
-
// System.out.println(entry.getKey()+":"+entry.getValue());
-
// }
-
originalExpiration = entrys.get("original-expiration").toString();
-
}
-
}
-
System.out.println(" [x] Received '" + envelope.getRoutingKey() + "':'" + message + "'" + "time" + System.currentTimeMillis());
-
HttpClient httpClient = new DefaultHttpClient();
-
HttpPost post = new HttpPost(message);
-
HttpResponse response = httpClient.execute(post);
-
BufferedReader inreader = null;
-
if (response.getStatusLine().getStatusCode() == 200) {
-
inreader = new BufferedReader(new InputStreamReader(response.getEntity().getContent(), "UTF-8"));
-
StringBuffer responseBody = new StringBuffer();
-
String line = null;
-
while ((line = inreader.readLine()) != null) {
-
responseBody.append(line);
-
}
-
if (!responseBody.equals("success")) {
-
// putDeclre15s(message);
-
if (originalExpiration.equals("0")) {
-
putDeclreQueue(message, 3000, QU_declare_15S);
-
}
-
if (originalExpiration.equals("3000")) {
-
putDeclreQueue(message, 30000, QU_declare_15S);
-
}
-
if (originalExpiration.equals("30000")) {
-
putDeclreQueue(message, 60000, QU_declare_15S);
-
}
-
if (originalExpiration.equals("60000")) {
-
putDeclreQueue(message, 120000, QU_declare_15S);
-
}
-
if (originalExpiration.equals("120000")) {
-
putDeclreQueue(message, 180000, QU_declare_15S);
-
}
-
if (originalExpiration.equals("180000")) {
-
putDeclreQueue(message, 300000, QU_declare_15S);
-
}
-
if (originalExpiration.equals("300000")) {
-
// channel.basicConsume(QU_declare_300S,true, this);
-
System.out.println("finish notify");
-
}
-
}
-
} else {
-
System.out.println(response.getStatusLine().getStatusCode());
-
}
-
}
-
};
-
-
channel.basicConsume(queueName, true, consumer);
-
} catch (Exception e) {
-
e.printStackTrace();
-
} finally {
-
}
-
}
-
-
-
-
static Map<String, Object> xdeathMap = new HashMap<String, Object>();
-
static List<Map<String, Object>> xDeath = new ArrayList<Map<String, Object>>();
-
static Map<String, Object> xdeathParam = new HashMap<String, Object>();
-
-
public static void putDeclre15s(String message) throws IOException {
-
channel.exchangeDeclare(EX_declare_15S, "topic");
-
Map<String, Object> args = new HashMap<String, Object>();
-
args.put("x-dead-letter-exchange", EXCHANGE_NAME);// 死信exchange
-
AMQP.BasicProperties.Builder builder = new AMQP.BasicProperties.Builder();
-
builder.expiration("3000").deliveryMode(2);// 设置消息TTL
-
AMQP.BasicProperties properties = builder.build();
-
channel.queueDeclare(QU_declare_15S, false, false, false, args);
-
channel.queueBind(QU_declare_15S, EX_declare_15S, ROUTINGKEY);
-
channel.basicPublish(EX_declare_15S, ROUTINGKEY, properties, message.getBytes());
-
System.out.println("send message in QA_DEFERRED_15S" + message + "time" + System.currentTimeMillis());
-
}
-
-
public static void putDeclreQueue(String message, int mis, String queue) throws IOException {
-
channel.exchangeDeclare(EX_declare_15S, "topic");
-
Map<String, Object> args = new HashMap<String, Object>();
-
args.put("x-dead-letter-exchange", EXCHANGE_NAME);// 死信exchange
-
AMQP.BasicProperties.Builder builder = new AMQP.BasicProperties.Builder();
-
builder.expiration(String.valueOf(mis)).deliveryMode(2);// 设置消息TTL
-
AMQP.BasicProperties properties = builder.build();
-
channel.queueDeclare(queue, false, false, false, args);
-
channel.queueBind(queue, EX_declare_15S, ROUTINGKEY);
-
channel.basicPublish(EX_declare_15S, ROUTINGKEY, properties, message.getBytes());
-
System.out.println("send message in " + queue + message + "time============" + System.currentTimeMillis());
-
}
-
-
public static void main(String args[]) throws Exception {
-
init();
-
consume();
-
}
-
}
消息通过dlx转发的情况下,header头部会带有x-death的一个数组,里面包含消息的各项属性,比如说消息成为死信的原因reason,original-expiration这个字段表示消息在原来队列中的过期时间,根据这个值来确定下一次通知的延迟时间应该是多少秒。
RabbitMQ 延迟队列实现订单支付结果异步阶梯性通知的更多相关文章
- C# RabbitMQ延迟队列功能实战项目演练
一.需求背景 当用户在商城上进行下单支付,我们假设如果8小时没有进行支付,那么就后台自动对该笔交易的状态修改为订单关闭取消,同时给用户发送一份邮件提醒.那么我们应用程序如何实现这样的需求场景呢?在之前 ...
- RabbitMQ延迟队列
rabbitmq延迟队列 rabbitmq实现延迟队列用了rabbitmq-delayed-message-exchange插件,需要提前安装,并启用. 原理 其原理是通过Exchange来实现延迟功 ...
- RabbitMQ延迟队列插件安装
RabbitMQ延迟队列插件安装 一.下载插件 下载地址:https://www.rabbitmq.com/community-plugins.html 二.把下载的插件放到指定位置 下载的文件为zi ...
- C#实现rabbitmq 延迟队列功能
最近在研究rabbitmq,项目中有这样一个场景:在用户要支付订单的时候,如果超过30分钟未支付,会把订单关掉.当然我们可以做一个定时任务,每个一段时间来扫描未支付的订单,如果该订单超过支付时间就关闭 ...
- Spring Boot(十四)RabbitMQ延迟队列
一.前言 延迟队列的使用场景:1.未按时支付的订单,30分钟过期之后取消订单:2.给活跃度比较低的用户间隔N天之后推送消息,提高活跃度:3.过1分钟给新注册会员的用户,发送注册邮件等. 实现延迟队列的 ...
- RabbitMQ 延迟队列,消息延迟推送
目录 应用场景 消息延迟推送的实现 测试结果 应用场景 目前常见的应用软件都有消息的延迟推送的影子,应用也极为广泛,例如: 淘宝七天自动确认收货.在我们签收商品后,物流系统会在七天后延时发送一个消息给 ...
- 【RabbitMQ】一文带你搞定RabbitMQ延迟队列
本文口味:鱼香肉丝 预计阅读:10分钟 一.说明 在上一篇中,介绍了RabbitMQ中的死信队列是什么,何时使用以及如何使用RabbitMQ的死信队列.相信通过上一篇的学习,对于死信队列已经有了更 ...
- SpringBoot RabbitMQ 延迟队列代码实现
场景 用户下单后,如果30min未支付,则删除该订单,这时候就要可以用延迟队列 准备 利用rabbitmq_delayed_message_exchange插件: 首先下载该插件:https://ww ...
- rabbitmq延迟队列demo
1. demo详解 1.1 工程结构: 1.2 pom 定义jar包依赖的版本.版本很重要,rabbit依赖spring,两者必须相一致,否则报错: <properties> <sp ...
随机推荐
- 洛谷—— P1434 滑雪
https://www.luogu.org/problem/show?pid=1434#sub 题目描述 Michael喜欢滑雪.这并不奇怪,因为滑雪的确很刺激.可是为了获得速度,滑的区域必须向下倾斜 ...
- 全然用linux工作,放弃windows
按: 虽然我们已经不习惯看长篇大论, 但我还是要说, 这是一篇值得你从头读到尾的长篇文章. 2005年9月22日,清华在读博士生王垠在水木社区BLOG上发表了<清华梦的粉碎--写给清华大学的退学 ...
- c++ builder firemonkey 实现填充椭圆
相信同类Delphi 类似文章非常多了,这里我用c++ builder firemonkey 实现填充椭圆 本例主要在FormPaint实现,当然你想在Image1->Bitmap->Ca ...
- Altium Designer的pcb界面如何让线变成点
但是16版本,需要tools --- Grid Manager --双击 双击后: 转自:http://blog.csdn.net/ldcung/article/details/77411434
- HDU 2473 Junk-Mail Filter 并查集删除(FZU 2155盟国)
http://acm.hdu.edu.cn/showproblem.php?pid=2473 http://acm.fzu.edu.cn/problem.php?pid=2155 题目大意: 编号0~ ...
- 从反编译深入理解JAVA内部类类结构以及finalkeyword
1.为什么成员内部类能够无条件訪问外部类的成员? 在此之前,我们已经讨论过了成员内部类能够无条件訪问外部类的成员,那详细到底是怎样实现的呢?以下通过反编译字节码文件看看到底.其实,编译器在进行编译的时 ...
- UVA10006 - Carmichael Numbers(筛选构造素数表+高速幂)
UVA10006 - Carmichael Numbers(筛选构造素数表+高速幂) 题目链接 题目大意:假设有一个合数.然后它满足随意大于1小于n的整数a, 满足a^n%n = a;这种合数叫做Ca ...
- Asp.NETCore让FromServices回来
起因 这两天,我忽然有点怀念 Asp.NET MVC 5 之前的时代,原因是我看到项目里面有这么一段代码(其实不止一段,几乎每个 Controller 都是) [Route("home&qu ...
- ORA-00119: invalid specification for system parameter LOCAL_LISTENER;
错误提示内容及上下文环境: SQL> grant sysdba to weng;grant sysdba to weng*第 1 行出现错误:ORA-01034: ORACLE not avai ...
- Redis Cluster(Redis 3.X)设计要点
Redis 3.0.0 RC1版本号10.9号公布,Release Note这个版本号支持Redis Cluster.相信非常多同学期待已久,只是这个版本号仅仅是RC版本号,要应用到生产环境,还得等等 ...