公众号开发-2种消息重排方案

公众号开发-2种消息重排方案

在进行微信公众号开发时,经常会发现接收到用户重复消息那是为什么呢,这是因为如果我们的服务没有在5秒内返回消息给微信服务器,微信服务器会断开我们的连接并且重新发起请求总共3次,这样会增大服务器的压力
本专栏说明以及源码地址:微信开发专栏说明

在这里插入图片描述

前言

本专栏所有代码都会提交到git中大家可以下载,别忘记点一个start哦

gitHub仓库:https://github.com/JolyouLu/JL-Public-Platform.git
gitee仓库:https://gitee.com/smallpage/jl-public-platform.git

解决思路

经过观察我们发现重复的消息都有一个相同的特点,就是内容都是重复的,那么我们只需要将接收到的用户消息保存下来,每一次收到消息就去检查一下是否是已保存过了,如果是就丢弃,如果不是就处理请求

那么保存什么字段呢?微信官方推荐是MsgId但是这有一个问题就是,事件消息是没有MsgId的,所以这里我使用的是CreateTime+FromUserName

怎么连接用户请求呢?当然是利用SpringAop面向切面编程拉,如果对SpringAop不是很连接可以阅读我的这篇文章Spring AOP原理深度解析-架构学习笔记14

在这里插入图片描述

代码实现

SpringAop+Map

利用SpringAop+Map适合单机服务,或没有部署Redis的项目

首先准备一个切面,一个HashMap

编写切入点,对切点使用环绕通知,在接口接收到用户消息时,获取消息中的FromUserName+CreateTime判断是否存在map中,不存在保存到map中就执行业务代码,如果在期间再次收到用户消息获取FromUserName+CreateTime发现已存在就会不做任何处理,当业务代码执行完毕后,将结果返回即可

@Slf4j
@Aspect
@Component
public class VxMsgAspect {

    //线程安全的HashMap
    private static final ConcurrentHashMap<String,String> map = new ConcurrentHashMap<>();

    //定义切入点
    @Pointcut("execution(* top.jolyoulu.controller.VxController.receiver(..))")
    public void pointCutVXReceiver(){};

    //环绕通知
    @Around("pointCutVXReceiver()")
    public Object around(ProceedingJoinPoint pjp) throws Throwable {
        Object retVal = null;
        //从连接点中获取参数
        Object[] args = pjp.getArgs();
        String xml = (String) args[0];
        //将来获取到的参数格式化
        Map<String, String> msgMap = MessageUtil.string2Map(xml);
        String msgId = msgMap.get("FromUserName") + msgMap.get("CreateTime");
        //如果已经存在于map中不做任何处理
        if (map.containsKey(msgId)){
            log.error("请求处理超时,接收到重复消息!");
            //直接返回空
            return null;
        }else {
            //否则将消息存入到map中
            map.put(msgId,msgId);
            //执行切面的方法,得到接口返回结果
            retVal = pjp.proceed(args);
            //执行完毕后删除map中保存的值
            map.remove(msgId);
        }
        //返回执行结果
        return retVal;
    }
}

SpringAop+Redis

利用SpringAop+Redis适合分布式项目,或有了部署Redis的项目
首先准备一个切面,一个Redis
编写切入点,对切点使用环绕通知,在接口接收到用户消息时,获取消息中的FromUserName+CreateTime判断是否存在redis中,不存在保存到redis中就执行业务代码,如果在期间再次收到用户消息获取>FromUserName+CreateTime发现已存在就会不做任何处理,当业务代码执行完毕后,将结果返回即可

@Slf4j
@Aspect
@Component
public class VxMsgAspect {

    //定义切入点
    @Pointcut("execution(* top.jolyoulu.controller.VxController.receiver(..))")
    public void pointCutVXReceiver(){};

    @Autowired
    private RedisCacheUtils redisCacheUtils;

    //环绕通知
    @Around("pointCutVXReceiver()")
    public Object before(ProceedingJoinPoint pjp) throws Throwable {
        Object retVal = null;
        //从连接点中获取参数
        Object[] args = pjp.getArgs();
        String xml = (String) args[0];
        //将来获取到的参数格式化
        Map<String, String> msgMap = MessageUtil.string2Map(xml);
        String msgId = msgMap.get("FromUserName") + msgMap.get("CreateTime");
        if (redisCacheUtils.existsCacheObj(msgId)){
            log.error("请求处理超时,接收到重复消息!");
            //直接返回空
            return null;
        }else {
            redisCacheUtils.setExCacheObj(msgId,msgId,10);
            //执行切面的方法,得到接口返回结果
            retVal = pjp.proceed(args);
        }
        return retVal;
    }
}

测试

在这里插入图片描述

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
根据题目要求,我们需要将给定的单链表重新排列为Ln→L1→Ln−1→L2→⋯的形式。可以使用两个数组来实现这个目标。首先,我们可以遍历链表,将链表中的每个节点的地址按顺序存储到一个数组中。接下来,我们可以利用这个数组来构建我们需要的排列后的链表。具体步骤如下: 1. 定义一个结构体,用于存储链表节点的数值和下一个节点的地址。 2. 遍历链表,将每个节点的地址按顺序存储到一个数组中。 3. 定义一个新的数组,利用第一个数组中的节点地址来构建排列后的链表。具体方法是,按照Ln→L1→Ln−1→L2→⋯的顺序,依次将第一个数组中的节点地址赋值给新数组中的每个元素。 4. 按照排列后的链表顺序,依次输出新数组中的节点数值即可。 总结一下,我们可以按照上述方法实现题目要求的重排链表的功能。引用提供了一个类似的思路,可以作为参考。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [L2-022 重排链表 【STL】](https://blog.csdn.net/wangws_sb/article/details/88644986)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* *3* [L2-022 重排链表](https://blog.csdn.net/weixin_45897672/article/details/104703349)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

JolyouLu

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值