RabbitMQ统一消息处理

添加消息请求头

使用RabbitMQ进行异步通信时,可能不仅仅需要传递消息本身,有时需要包含一些额外的信息,比如登录状态。可以参考前端发送请求时的做法,把登录状态保存在请求头中。Message类中也有请求头,可以使用一下方式添加到请求头:

java 复制代码
message.getMessageProperties().getHeaders().put("user-info", 1L);

实际使用中,没有必要每次都创建一个Message类,然后手动添加。通过MessagePostProcessor可以简单实现请求头添加,而最常用的convertAndSend方法就可以直接传入MessagePostProcessor作为参数:

java 复制代码
String exchangeName = "test.direct";
String key = "test";
rabbitTemplate.convertAndSend(exchangeName, key, "msg", new MessagePostProcessor() {
    @Override
    public Message postProcessMessage(Message message) throws AmqpException {
        message.getMessageProperties().setHeader("user-info", 1L);
        // 这里也可以做一些其他的统一逻辑处理
        return message;
    }
});

也可以这样写:

java 复制代码
@Data
public class MyMessagePostProcessor implements MessagePostProcessor {
    
    private Long id;

    public MyMessagePostProcessor(Long id) {
        this.id = id;
    }

    @Override
    public Message postProcessMessage(Message message) throws AmqpException {
        message.getMessageProperties().getHeaders().put("user-info", id);
        return message;
    }
}
// 后续就不需要重写添加请求头的逻辑了
rabbitTemplate.convertAndSend(exchangeName, key, "msg", new MyMessagePostProcessor(id);

消费消息请求头

使用消息后置处理器

实际上,RabbitTemplate也可以设置前置/后置处理器:

java 复制代码
@PostConstruct// 初始化完成后立即执行
public void init(RabbitTemplate rabbitTemplate) {
    // 添加消息后置处理器(消费前执行)
    rabbitTemplate.setAfterReceivePostProcessors(new MessagePostProcessor() {
        @Override
        public Message postProcessMessage(Message message) throws AmqpException {
            Long userId = (Long) message.getMessageProperties().getHeaders().get("user-info");
            UserContext.setUser(userId);
            return message;
        }
    });
    // 添加消息前置处理器
    rabbitTemplate.setBeforePublishPostProcessors(new MessagePostProcessor() {
        @Override
        public Message postProcessMessage(Message message) throws AmqpException {
            message.getMessageProperties().getHeaders().put("user-info", 1L);
            return message;
        }
    });
}

通过这种方法就不需要再每次发送消息前加一个处理器了,然而这种方式并不十分可行,它所添加的后置处理器针对的是rabbitTemplate,也就是消费端必须通过rabbitTemplate(也必须是同一个rabbitTemplate)消费,RabbitTemplate提供了两个直接消费的方法:

  • receive() 阻塞等待一定时间,返回值为Message,超时返回null;
  • receiveNowait() 直接尝试获取对应queue中的消息,返回值为Message,如果没有消息返回null;

在异步场景下显然这两种方法都不满足要求。

一般情况下,消费更倾向于使用监听机制,也就是使用@RabbitListener注解,这种方式也可以自动的声明交换机、队列以及建立绑定,因此使用更加广泛

java 复制代码
@RabbitListener(bindings = @QueueBinding(
        value = @Queue(name = "test.queue", durable = "true"),
        exchange = @Exchange(name = "test.direct"),
        key = "test"
))
public void listenerTest(String message) {
    // todo
}

然而这种方式无法执行前面设置的后置处理器逻辑,也就是拿不到消息头中的信息。

手动获取或利用AOP

直接接收Message作为参数:

java 复制代码
@RabbitListener(bindings = @QueueBinding(
        value = @Queue(name = "test.queue", durable = "true"),
        exchange = @Exchange(name = "test.direct"),
        key = "test"
))
public void listenerTest(String msg, Message message) {
    // 这里不传消息体作为参数,MessageConvert就不会自动进行反序列化
    Long userId = (Long) message.getMessageProperties().getHeaders().get("user-info");
    System.out.println(userId);
}

因此也可以通过Spring AOP 把处理请求头的逻辑抽取出来(AOP增强的是方法本身,因此即使方法中用不到Message这个参数,也不可以省略,否则就找不到消息头了):

java 复制代码
// 定义切面类
@Aspect
@Component
public class RabbitListenerAspect {

    @Before("@annotation(org.springframework.amqp.rabbit.annotation.RabbitListener)")
    public void beforeMessageProcessing(JoinPoint joinPoint) {
        Object[] args = joinPoint.getArgs();

        for (Object arg : args) {
            if (arg instanceof Message) {
                Message message = (Message) arg;

                Map<String, Object> headers = message.getMessageProperties().getHeaders();
                Long userId = (Long) headers.get("user-info");
                UserContext.setUser(userId);
            }
        }
    }
}

利用消息转换器

使用convertAndSend方法发送消息时,在发送前会通过MessageConvert对消息进行序列化(默认使用jdk序列化),在消费者接收前,也会同样的调用对应转换器中的反序列化工具转化为Java对象,因此可以利用反序列化这个过程,由于反序列化的过程发生在消费者端,因此可以在这个过程中执行一些消费前的逻辑:

java 复制代码
// 这里使用的是Json序列化转换器
public class CustomMessageConverter extends Jackson2JsonMessageConverter {
    @Override
    public Object fromMessage(Message message) {
        // 获取headers
        Map<String, Object> headers = message.getMessageProperties().getHeaders();
        Long userId = (Long) headers.get("user-info");
        // 添加用户信息到上下文
        UserContext.setUser(userId);
        // 调用父类方法反序列化
        return super.fromMessage(message);
    }
}
相关推荐
魔道不误砍柴功14 分钟前
Java 中如何巧妙应用 Function 让方法复用性更强
java·开发语言·python
NiNg_1_23414 分钟前
SpringBoot整合SpringSecurity实现密码加密解密、登录认证退出功能
java·spring boot·后端
闲晨17 分钟前
C++ 继承:代码传承的魔法棒,开启奇幻编程之旅
java·c语言·开发语言·c++·经验分享
测开小菜鸟2 小时前
使用python向钉钉群聊发送消息
java·python·钉钉
P.H. Infinity3 小时前
【RabbitMQ】04-发送者可靠性
java·rabbitmq·java-rabbitmq
生命几十年3万天3 小时前
java的threadlocal为何内存泄漏
java
caridle3 小时前
教程:使用 InterBase Express 访问数据库(五):TIBTransaction
java·数据库·express
^velpro^3 小时前
数据库连接池的创建
java·开发语言·数据库
苹果醋33 小时前
Java8->Java19的初步探索
java·运维·spring boot·mysql·nginx
秋の花3 小时前
【JAVA基础】Java集合基础
java·开发语言·windows