前言
(防止自己以后再踩坑,这里记录一下)
这个异常是我在设置队列过期时间遇到的,搞了半天也查找了很多的资料大部分都是2种解决方案:
1、如果队列设置的是客户端是自动创建的,直接删除队列。
2、如果客户端没有配置自动创建队列的话,手动去MQ客户端创建队列,并且设置对应的TTL值。
异常以及解决方案
上面2种方法我都试过了,还是没有解决我的这个问题。后来经过不停的试验终于搞定了。
客户端自动创建队列(现在一般都是这种啦),设置x-message-ttl队列过期时间,但是不指定type(我就这种情况)
配置:
@RabbitListener(bindings = {
@QueueBinding(
value = @Queue(value = "direct.info", //指定一下队列名,默认持久队列,不指定则为临时队列
arguments = {
@Argument(name = "x-dead-letter-exchange",value = "dlx.exchange"), //指定一下死信交换机
@Argument(name = "x-dead-letter-routing-key",value = "deadKey"), //指定死信交换机的路由key
@Argument(name = "x-message-ttl",value = "3000") //指定队列的过期时间
}),
exchange = @Exchange(value = "directs"),//Exchang的默认类型就是direct,所以type可以不写
key = {"info"}
)
})
private void receive(Message message, Channel channel) throws Exception{
System.out.println(new String(message.getBody()));
}
启动客户端报错:
Caused by: com.rabbitmq.client.ShutdownSignalException: channel error; protocol method: #method<channel.close>(reply-code=406, reply-text=PRECONDITION_FAILED - invalid arg 'x-message-ttl' for queue 'direct.info' in vhost '/test': {unacceptable_type,longstr}, class-id=50, method-id=10)
注意红色字体标记出来的,大概意思就是不能接受的类型,但是这个longstr类型是个啥我也不知道,反Long和String都试一试呗,首先看看@Argument的参数
@Target({})
@Retention(RetentionPolicy.RUNTIME)
public @interface Argument {
String name();
String value() default "";
String type() default "java.lang.String";
}
如果type不指定时他默认的就是Sting类型的,那就换成Long试试看,修改一下代码
@Argument(name = "x-message-ttl",value = "3000",type = "java.lang.Long") //指定队列的过期时间
结果出乎我的意料,异常消失了........................队列成功创建
总结:这种异常就是需要指定x-message-ttl的type类型,不能使用默认的String,改成Long类型就好使了,我也不晓得为啥要这么干,既然String类型不好使为啥要默认String呢?
其他情况
其他的意外这里做个笔记
在mq客户端创建队列并且指定x-message-ttl时,类型需要选择Number,不能选择String,如果选择String会提示报错
当你选择Number并且成功创建了队列,但是如果@Argument(name = "x-message-ttl",value = "3000")这样写,不指定type类型而使用默认的话,会报错:
Caused by: com.rabbitmq.client.ShutdownSignalException: channel error; protocol method: #method<channel.close>(reply-code=406, reply-text=PRECONDITION_FAILED - inequivalent arg 'x-message-ttl' for queue 'direct.info' in vhost '/test': received '3000' but current is '3000', class-id=50, method-id=10)
我当时看到这个异常是一脸懵逼呀,MQ客户端设置的x-message-ttl明明是3000,@Argument(name = "x-message-ttl",value = "3000")设置的也是3000,为什么会报出这么个异常呢?归根结底还是需要在设置x-message-ttl时必须要指定一下type为java.lang.Long(默认的String都不行)