准备工作:

1)安装RabbitMQ,参考文章:消息中间件系列二:RabbitMQ入门(基本概念、RabbitMQ的安装和运行)

2.)分别新建名为OriginalRabbitMQProducer和OriginalRabbitMQConsumer的maven工程

在pom.xml文件里面引入如下依赖:

    <dependency>
<groupId>com.rabbitmq</groupId>
<artifactId>amqp-client</artifactId>
<version>5.0.0</version>
</dependency>

说明:5系列的版本最好使用JDK8及以上, 低于JDK8可以使用4.x(具体的版本号到Maven的中央仓库查)的版本

一、消费者(接收方)自动确认模式

 前面有谈到消费者收到的每一条消息都必须进行确认,消息的确认机制分为自动确认和消费者自行确认。下面我们来看一下自动确认的示例:

示例1:交换器是direct

1. 在工程OriginalRabbitMQProducer新建一个一个direct的生产者

package study.demo.normal;

import com.rabbitmq.client.BuiltinExchangeType;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory; import java.io.IOException;
import java.util.concurrent.TimeoutException; /**
*
* @Description: 交换器是direct的生产者 路由键完全匹配时,消息才投放到对应队列
* @author leeSmall
* @date 2018年9月15日
*
*/
public class DirectProducer { //定义交换器的名字
private final static String EXCHANGE_NAME = "direct_logs"; public static void main(String[] args) throws IOException, TimeoutException { //1.创建一个连接工厂
ConnectionFactory factory = new ConnectionFactory(); //设置要连接的RabbitMQ服务器的地址
factory.setHost("127.0.0.1"); //设置用户名 这里使用缺省的
//factory.setUsername(..); //设置连接断开 这里使用缺省的
//factory.setPort(); //设置虚拟主机 这里使用缺省的
//factory.setVirtualHost(); //2.通过连接工厂创建一个连接
Connection connection = factory.newConnection(); //3.通过连接创建一个信道 信道是用来传送数据的
Channel channel = connection.createChannel(); //4.通过信道声明一个交换器 第一个参数时交换器的名字 第二个参数时交换器的种类
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT); //定义一组路由键
String[]routingKeys = {"error","info","warning"}; //发布消息的交换器上
for(int i=0;i<3;i++){
//路由键
String routingKey = routingKeys[i]; //要发送的消息
String message = "Hello world_"+(i+1); /**
* 发送消息到交换器上
* 参数1:交换器的名字
* 参数2:路由键
* 参数3:BasicProperties
* 参数4:要发送的消息
*/
channel.basicPublish(EXCHANGE_NAME,routingKey,null,message.getBytes());
System.out.println("Sent "+routingKey+":"+message); } channel.close();
connection.close(); } }

2. 在工程OriginalRabbitMQConsumer新建一个direct的只消费error日志的消费者

package study.demo.normal;

import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException; /**
*
* @Description: 交换器是direct 只消费error日志的消费者
* @author leeSmall
* @date 2018年9月15日
*
*/
public class ConsumerError { //定义交换器的名字
private static final String EXCHANGE_NAME = "direct_logs";
// private static final String EXCHANGE_NAME = "fanout_logs_1"; public static void main(String[] argv) throws IOException, TimeoutException {
//1.创建一个连接工厂
ConnectionFactory factory = new ConnectionFactory();
//设置要连接的RabbitMQ服务器的地址
factory.setHost("127.0.0.1"); //2.通过连接工厂创建一个连接
Connection connection = factory.newConnection(); //3.通过连接创建一个信道 信道是用来传送数据的
Channel channel = connection.createChannel(); //4.通过信道声明一个交换器 第一个参数时交换器的名字 第二个参数时交换器的种类
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT); //5.声明随机队列
String queueName = channel.queueDeclare().getQueue(); //6.声明一个只消费错误日志的路由键error
String routingKey = "error"; //7.队列通过路由键绑定到交换器上
channel.queueBind(queueName,EXCHANGE_NAME,routingKey); System.out.println("Waiting message......."); //8.设置一个监听器监听消费消息
Consumer consumerB = 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");
System.out.println("Accept:"+envelope.getRoutingKey()+":"+message);
}
};
     //9.自动确认:autoAck参数为true
channel.basicConsume(queueName,true,consumerB);
} }

3. 启动消费者ConsumerError,再启动生产者DirectProducer,查看效果

启动消费者ConsumerError:

启动生产者DirectProducer:

查看消费者ConsumerError现在的状况:

可以看到消费者只消费了error级别的消息,这是因为在direct模式下,消费者只定义了路由键routingKey = "error";

4. 在工程OriginalRabbitMQConsumer新建一个direct的消费所有日志的消费者

package study.demo.normal;

import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException; /**
*
* @Description: 交换器是direct 消费所有日志的消费者
* @author leeSmall
* @date 2018年9月15日
*
*/
public class ConsumerAll {
//定义交换器的名字
private static final String EXCHANGE_NAME = "direct_logs";
// private static final String EXCHANGE_NAME = "fanout_logs_1"; public static void main(String[] argv) throws IOException,
InterruptedException, TimeoutException { //1.创建一个连接工厂
ConnectionFactory factory = new ConnectionFactory();
//设置要连接的RabbitMQ服务器的地址
factory.setHost("127.0.0.1"); //2.通过连接工厂创建一个连接
Connection connection = factory.newConnection(); //3.通过连接创建一个信道 信道是用来传送数据的
Channel channel = connection.createChannel(); //4.通过信道声明一个交换器 第一个参数时交换器的名字 第二个参数时交换器的种类
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT); //5.声明随机队列
String queueName = channel.queueDeclare().getQueue(); //6.定义一组路由键消费所有日志
String[]routingKeys = {"error","info","warning"}; //7.队列通过路由键绑定到交换器上
for(String routingKey:routingKeys){
//队列和交换器的绑定
channel.queueBind(queueName,EXCHANGE_NAME,routingKey);
}
System.out.println("Waiting message......."); //8.设置一个监听器监听消费消息
Consumer consumerA = 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");
System.out.println("Accept:"+envelope.getRoutingKey()+":"+message);
}
};
//9.自动确认:autoAck参数为true
channel.basicConsume(queueName,true,consumerA);     }   }

5. 启动消费者ConsumerAll,再启动生产者DirectProducer,查看效果

启动消费者ConsumerAll:

启动生产者DirectProducer:

查看消费者ConsumerAll的状态:

 可以看到消费者ConsumerAll消费了生产者DirectProducer产生的所有消息,这是因为在direct模式下,消费者定义了和生产者一样个数的路由键String[]routingKeys = {"error","info","warning"};

示例2:交换器是fanout

1. 在工程OriginalRabbitMQProducer新建一个交换器是fanout的生产者

package study.demo.normal;

import com.rabbitmq.client.BuiltinExchangeType;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory; import java.io.IOException;
import java.util.concurrent.TimeoutException; /**
*
* @Description: 交换器是fanout的生产者 可以理解为广播,会把所有消息投放到绑定到这个交换器上的队列上
* @author leeSmall
* @date 2018年9月15日
*
*/
public class FanoutProducer { private final static String EXCHANGE_NAME = "fanout_logs_1"; public static void main(String[] args) throws IOException, TimeoutException { //1.创建一个连接工厂
ConnectionFactory factory = new ConnectionFactory(); //设置要连接的RabbitMQ服务器的地址
factory.setHost("127.0.0.1"); //设置用户名 这里使用缺省的
//factory.setUsername(..); //设置连接断开 这里使用缺省的
//factory.setPort(); //设置虚拟主机 这里使用缺省的
//factory.setVirtualHost(); //2.通过连接工厂创建一个连接
Connection connection = factory.newConnection(); //3.通过连接创建一个信道 信道是用来传送数据的
Channel channel = connection.createChannel(); //4.通过信道声明一个交换器 第一个参数时交换器的名字 第二个参数时交换器的种类
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.FANOUT); //定义一组路由键
String[]routingKeys = {"error","info","warning"}; //发布消息的交换器上
for(int i=0;i<3;i++){
//路由键
String routingKey = routingKeys[i]; //要发送的消息
String message = "Hello world_"+(i+1); /**
* 发送消息到交换器上
* 参数1:交换器的名字
* 参数2:路由键
* 参数3:BasicProperties
* 参数4:要发送的消息
*/
channel.basicPublish(EXCHANGE_NAME,routingKey,null,message.getBytes());
System.out.println("Sent "+routingKey+":"+message); } channel.close();
connection.close(); } }

2. 在工程OriginalRabbitMQConsumer新建一个fanout的只消费error日志的消费者

package study.demo.normal;

import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException; /**
*
* @Description: 交换器是fanout,只消费error日志的消费者
* @author leeSmall
* @date 2018年9月15日
*
*/
public class ConsumerError { //定义交换器的名字
//private static final String EXCHANGE_NAME = "direct_logs";
private static final String EXCHANGE_NAME = "fanout_logs_1"; public static void main(String[] argv) throws IOException, TimeoutException {
//1.创建一个连接工厂
ConnectionFactory factory = new ConnectionFactory();
//设置要连接的RabbitMQ服务器的地址
factory.setHost("127.0.0.1"); //2.通过连接工厂创建一个连接
Connection connection = factory.newConnection(); //3.通过连接创建一个信道 信道是用来传送数据的
Channel channel = connection.createChannel(); //4.通过信道声明一个交换器 第一个参数时交换器的名字 第二个参数时交换器的种类
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.FANOUT); //5.声明随机队列
String queueName = channel.queueDeclare().getQueue(); //6.声明一个只消费错误日志的路由键error
String routingKey = "error"; //7.队列通过路由键绑定到交换器上
channel.queueBind(queueName,EXCHANGE_NAME,routingKey); System.out.println("Waiting message......."); //8.设置一个监听器监听消费消息
Consumer consumerB = 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");
System.out.println("Accept:"+envelope.getRoutingKey()+":"+message);
}
};
     //9.自动确认:autoAck参数为true
     channel.basicConsume(queueName,true,consumerA); } }

3. 启动消费者ConsumerError,再启动生产者DirectProducer,查看效果

启动消费者ConsumerError:

启动生产者FanoutProducer:

查看消费者ConsumerError现在的状况:

可以看到消费者消费了所有级别的消息,这是因为在fanout模式下,虽然消费者只定义了路由键routingKey = "error",但是因为fanut是广播模式,会把所有消息投放到绑定到这个交换器上的队列上

4. 在工程OriginalRabbitMQConsumer新建一个fanout的消费所有日志的消费者

package study.demo.normal;

import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException; /**
*
* @Description: 交换器是fanout 消费所有日志的消费者
* @author leeSmall
* @date 2018年9月15日
*
*/
public class ConsumerAll {
//定义交换器的名字
//private static final String EXCHANGE_NAME = "direct_logs";
private static final String EXCHANGE_NAME = "fanout_logs_1"; public static void main(String[] argv) throws IOException,
InterruptedException, TimeoutException { //1.创建一个连接工厂
ConnectionFactory factory = new ConnectionFactory();
//设置要连接的RabbitMQ服务器的地址
factory.setHost("127.0.0.1"); //2.通过连接工厂创建一个连接
Connection connection = factory.newConnection(); //3.通过连接创建一个信道 信道是用来传送数据的
Channel channel = connection.createChannel(); //4.通过信道声明一个交换器 第一个参数时交换器的名字 第二个参数时交换器的种类
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.FANOUT); //5.声明随机队列
String queueName = channel.queueDeclare().getQueue(); //6.定义一组路由键消费所有日志
String[]routingKeys = {"error","info","warning"}; //7.队列通过路由键绑定到交换器上
for(String routingKey:routingKeys){
//队列和交换器的绑定
channel.queueBind(queueName,EXCHANGE_NAME,routingKey);
}
System.out.println("Waiting message......."); //8.设置一个监听器监听消费消息
Consumer consumerA = 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");
System.out.println("Accept:"+envelope.getRoutingKey()+":"+message);
}
};
     //9.自动确认:autoAck参数为true
     channel.basicConsume(queueName,true,consumerA); }
}

5. 启动消费者ConsumerAll,再启动生产者DirectProducer,查看效果

启动消费者ConsumerAll:

启动生产者FanoutProducer:

查看消费者ConsumerAll的状态:

 可以看到消费者ConsumerAll消费了生产者DirectProducer产生的所有消息,这是因为在fanout模式下,消费者定义了和生产者一样个数的路由键String[]routingKeys = {"error","info","warning"};

三、消费者(接收方)自行确认模式

1. 在工程OriginalRabbitMQProducer新建一个消费者自行确认生产者

package study.demo.consumerconfirm;

import com.rabbitmq.client.BuiltinExchangeType;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory; import java.io.IOException;
import java.util.concurrent.TimeoutException; /**
*
* @Description: 消费者自行确认生产者
* @author leeSmall
* @date 2018年9月15日
*
*/
public class ConsumerConfirmProducer { //交换器
private final static String EXCHANGE_NAME = "direct_cc_confirm_1"; //路由键
private final static String ROUTE_KEY = "error"; public static void main(String[] args) throws IOException, TimeoutException,
InterruptedException { //1.创建一个连接工厂
ConnectionFactory factory = new ConnectionFactory(); //设置要连接的RabbitMQ服务器的地址
factory.setHost("127.0.0.1"); //设置用户名 这里使用缺省的
//factory.setUsername(..); //设置连接断开 这里使用缺省的
//factory.setPort(); //设置虚拟主机 这里使用缺省的
//factory.setVirtualHost(); //2.通过连接工厂创建一个连接
Connection connection = factory.newConnection(); //3.通过连接创建一个信道 信道是用来传送数据的
Channel channel = connection.createChannel(); //4.通过信道声明一个交换器 第一个参数时交换器的名字 第二个参数时交换器的种类
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT); //发布消息的交换器上
for(int i=0;i<10;i++){
//要发送的消息
String message = "Hello world_"+(i+1); /**
* 发送消息到交换器上
* 参数1:交换器的名字
* 参数2:路由键
* 参数3:BasicProperties
* 参数4:要发送的消息
*/
channel.basicPublish(EXCHANGE_NAME,ROUTE_KEY,null,message.getBytes());
System.out.println("Sent "+ROUTE_KEY+":"+message); } channel.close();
connection.close(); } }

2. 在工程OriginalRabbitMQConsumer新建一个消费者自行确认消费者

package study.demo.consumerconfirm;

import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException; /**
*
* @Description: 消费者自行确认消费者
* @author leeSmall
* @date 2018年9月15日
*
*/
public class ClientConsumerAck { private static final String EXCHANGE_NAME = "direct_cc_confirm_1"; public static void main(String[] argv) throws IOException, TimeoutException {
//1.创建一个连接工厂
ConnectionFactory factory = new ConnectionFactory();
//设置要连接的RabbitMQ服务器的地址
factory.setHost("127.0.0.1"); //2.通过连接工厂创建一个连接
Connection connection = factory.newConnection(); //3.通过连接创建一个信道 信道是用来传送数据的
Channel channel = connection.createChannel(); //4.通过信道声明一个交换器 第一个参数时交换器的名字 第二个参数时交换器的种类
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT); //5.声明队列
String queueName = "consumer_confirm";
channel.queueDeclare(queueName,false,false,
false,null); //6.声明一个只消费错误日志的路由键error
String routingKey = "error"; //7.队列通过路由键绑定到交换器上
channel.queueBind(queueName,EXCHANGE_NAME,routingKey);
System.out.println("Waiting message......."); //8.设置一个监听器监听消费消息
Consumer consumerB = 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");
System.out.println("Accept:"+envelope.getRoutingKey()+":"+message);
//消费者自行确认
this.getChannel().basicAck(envelope.getDeliveryTag(),false);
}
}; //9.消费者自行确认:autoAck参数为false
channel.basicConsume(queueName,false,consumerB);
} } 

3. 在工程OriginalRabbitMQConsumer新建一个消费者自行确认休眠不回复ack的消费者

package study.demo.consumerconfirm;

import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException; /**
*
* @Description: 消费者自行确认休眠不回复ack的消费者
* @author leeSmall
* @date 2018年9月15日
*
*/
public class ClientConsumerSlowAck { private static final String EXCHANGE_NAME = "direct_cc_confirm_1"; public static void main(String[] argv) throws IOException, TimeoutException {
//1.创建一个连接工厂
ConnectionFactory factory = new ConnectionFactory();
//设置要连接的RabbitMQ服务器的地址
factory.setHost("127.0.0.1"); //2.通过连接工厂创建一个连接
Connection connection = factory.newConnection(); //3.通过连接创建一个信道 信道是用来传送数据的
Channel channel = connection.createChannel(); //4.通过信道声明一个交换器 第一个参数时交换器的名字 第二个参数时交换器的种类
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT); //5.声明队列
String queueName = "consumer_confirm";
channel.queueDeclare(queueName,false,false,
false,null); //6.声明一个只消费错误日志的路由键error
String routingKey = "error"; //7.队列通过路由键绑定到交换器上
channel.queueBind(queueName,EXCHANGE_NAME,routingKey);
System.out.println("Waiting message......."); //8.设置一个监听器监听消费消息
Consumer consumerB = new DefaultConsumer(channel){
@Override
public void handleDelivery(String consumerTag, Envelope envelope,
AMQP.BasicProperties properties, byte[] body)
throws IOException {
try {
//消费者自行确认时不回复ack,一直休眠
Thread.sleep(20000);
} catch (InterruptedException e) {
e.printStackTrace();
}
String message = new String(body,"UTF-8");
System.out.println("Accept:"+envelope.getRoutingKey()+":"+message);
//this.getChannel().basicAck(envelope.getDeliveryTag(),false);
}
}; //9.消费者自行确认:autoAck参数为false
channel.basicConsume(queueName,false,consumerB);
} } 

4. 分别启动消费者自行确认消费者ClientConsumerAck和消费者自行确认休眠不回复ack的消费者ClientConsumerSlowAck,再启动消费者自行确认生产者ConsumerConfirmProducer查看状态

启动消费者自行确认消费者ClientConsumerAck:

启动消费者自行确认休眠不回复ack的消费者ClientConsumerSlowAck

启动消费者自行确认生产者ConsumerConfirmProducer

查看消费者自行确认消费者ClientConsumerAck的状态:

查看消费者自行确认休眠不回复ack的消费者ClientConsumerSlowAck的状态:

查看RabbitMQ服务器上的队列情况:

可以看到队列consumer_confirm里面有5条消息未消费,这是因为消费者自行确认休眠不回复ack的消费者ClientConsumerSlowAck收到了这5条消息,但是没有向RabbitMQ服务器发送确认消息,RabbitMQMQ认为这5条消息还没有被消费就一直存在队列里面

下面停掉ClientConsumerSlowAck,查看ClientConsumerAck和RabbitMQ服务器里面队列consumer_confirm的状态

可以看到停掉ClientConsumerSlowAck以后,之前的5条消息被ClientConsumerAck消费了

5. 在工程OriginalRabbitMQConsumer 新建一个消费者自行确认拒绝消息的消费者

package study.demo.consumerconfirm;

import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException; /**
*
* @Description: 消费者自行确认拒绝消息的消费者
* @author leeSmall
* @date 2018年9月15日
*
*/
public class ClientConsumerReject { private static final String EXCHANGE_NAME = "direct_cc_confirm_1"; public static void main(String[] argv) throws IOException, TimeoutException {
//1.创建一个连接工厂
ConnectionFactory factory = new ConnectionFactory();
//设置要连接的RabbitMQ服务器的地址
factory.setHost("127.0.0.1"); //2.通过连接工厂创建一个连接
Connection connection = factory.newConnection(); //3.通过连接创建一个信道 信道是用来传送数据的
Channel channel = connection.createChannel(); //4.通过信道声明一个交换器 第一个参数时交换器的名字 第二个参数时交换器的种类
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT); //5.声明队列
String queueName = "consumer_confirm";
channel.queueDeclare(queueName,false,false,
false,null); //6.声明一个只消费错误日志的路由键error
String routingKey = "error"; //7.队列通过路由键绑定到交换器上
channel.queueBind(queueName,EXCHANGE_NAME,routingKey);
System.out.println("Waiting message......."); //8.设置一个监听器监听消费消息
Consumer consumerB = new DefaultConsumer(channel){
@Override
public void handleDelivery(String consumerTag, Envelope envelope,
AMQP.BasicProperties properties, byte[] body)
throws IOException {
//消费者自行拒绝消息 参数requeue=true,让RabbitMQ服务器重新分发消息,requeue=false让RabbitMQ服务器移除消息
this.getChannel().basicReject(envelope.getDeliveryTag(),true);
System.out.println("Reject:"+envelope.getRoutingKey()
+":"+new String(body,"UTF-8"));
}
}; //9.消费者自行确认:autoAck参数为false
channel.basicConsume(queueName,false,consumerB);
} } 

6. 分别启动消费者自行确认消费者ClientConsumerAck和消费者自行确认拒绝消息的消费者ClientConsumerReject,再启动消费者自行确认生产者ConsumerConfirmProducer查看状态

启动消费者自行确认消费者ClientConsumerAck:

启动消费者自行确认拒绝消息的消费者ClientConsumerReject:

启动消费者自行确认生产者ConsumerConfirmProducer

查看消费者自行确认消费者ClientConsumerAck的状态:

查看消费者自行确认拒绝消息的消费者ClientConsumerReject的状态:

可以看到消息都被ClientConsumerAck消费了,这是因为消费者ClientConsumerReject拒绝了所有消息,这里要注意

this.getChannel().basicReject(envelope.getDeliveryTag(),true);

这段代码的basicReject的第二个参数requeue,参数requeue=true,让RabbitMQ服务器重新分发消息,requeue=false让RabbitMQ服务器移除消息

 requeue=false时,RabbitMQ服务器会删掉被ClientConsumerReject拒绝的消息,消费者ClientConsumerAck就不能消费所有消息了

四、生产者(发送方)确认模式

为什么要有个发送方确认模式?

生产者不知道消息是否真正到达RabbitMq,也就是说发布操作不返回任何消息给生产者。
AMQP协议层面为我们提供的事务机制解决了这个问题,但是事务机制本身也会带来问题:
1)严重的性能问题
2)使生产者应用程序产生同步
RabbitMQ团队为我们拿出了更好的方案,即采用发送方确认模式,该模式比事务更轻量,性能影响几乎可以忽略不计。

1. 在OriginalRabbitMQProducer工程新建一个生产者(发送方)确认同步模式的类

package study.demo.producerconfirm;

import java.io.IOException;
import java.util.concurrent.TimeoutException; import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory; /**
*
* @Description: 生产者(发送方)确认同步模式
* @author leeSmall
* @date 2018年9月16日
*
*/
public class ProducerConfirm { private final static String EXCHANGE_NAME = "producer_confirm";
private final static String ROUTE_KEY = "error"; public static void main(String[] args) throws IOException, TimeoutException,
InterruptedException {
//1.创建一个连接工厂
ConnectionFactory factory = new ConnectionFactory(); //设置要连接的RabbitMQ服务器的地址
factory.setHost("127.0.0.1"); //设置用户名 这里使用缺省的
//factory.setUsername(..); //设置连接断开 这里使用缺省的
//factory.setPort(); //设置虚拟主机 这里使用缺省的
//factory.setVirtualHost(); //2.通过连接工厂创建一个连接
Connection connection = factory.newConnection(); //3.通过连接创建一个信道 信道是用来传送数据的
Channel channel = connection.createChannel();
//将信道设置为发送方确认
channel.confirmSelect(); //发布消息的交换器上
for(int i=0;i<2;i++){
String msg = "Hello "+(i+1);
channel.basicPublish(EXCHANGE_NAME,ROUTE_KEY,null,msg.getBytes());
//等待RabbitMQ返回消息确认消息已送达RabbitMQ服务器
if (channel.waitForConfirms()){
System.out.println("发送方同步确认: "+ROUTE_KEY+":"+msg);
}
} // 关闭频道和连接
channel.close();
connection.close();
} }

2. 在OriginalRabbitMQProducer工程新建一个生产者(发送方)确认异步模式的类

package study.demo.producerconfirm;

import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException; /**
*
* @Description: 生产者(发送方)确认异步模式
* @author leeSmall
* @date 2018年9月16日
*
*/
public class ProducerConfirmAsync { private final static String EXCHANGE_NAME = "producer_confirm"; public static void main(String[] args) throws IOException, TimeoutException,
InterruptedException {
//1.创建一个连接工厂
ConnectionFactory factory = new ConnectionFactory(); //设置要连接的RabbitMQ服务器的地址
factory.setHost("127.0.0.1"); //设置用户名 这里使用缺省的
//factory.setUsername(..); //设置连接断开 这里使用缺省的
//factory.setPort(); //设置虚拟主机 这里使用缺省的
//factory.setVirtualHost(); //2.通过连接工厂创建一个连接
Connection connection = factory.newConnection(); //3.通过连接创建一个信道 信道是用来传送数据的
Channel channel = connection.createChannel(); //4.通过信道声明一个交换器 第一个参数时交换器的名字 第二个参数时交换器的种类
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT); //将信道设置为发送方确认
channel.confirmSelect(); //信道被关闭监听器 用于RabbitMQ服务器断线重连
//channel.addShutdownListener(); /**
* 生产者异步确认监听
* 参数deliveryTag代表了当前channel唯一的投递
* 参数multiple:false
*
*/
channel.addConfirmListener(new ConfirmListener() {
//RabbitMQ服务器确认收到消息
public void handleAck(long deliveryTag, boolean multiple)
throws IOException {
System.out.println("RabbitMQ服务器确认收到消息Ack deliveryTag="+deliveryTag
+"multiple:"+multiple);
} //RabbitMQ服务器由于自己内部出现故障没有收到消息
public void handleNack(long deliveryTag, boolean multiple)
throws IOException {
System.out.println("RabbitMQ服务没有收到消息Ack deliveryTag="+deliveryTag
+"multiple:"+multiple);
}
}); //生产者异步返回监听 这里和发布消息时的mandatory参数有关
//参数mandatory:mandatory=true,投递消息时无法找到一个合适的队列,把消息返回给生产者,mandatory=false 丢弃消息(缺省)
channel.addReturnListener(new ReturnListener() {
public void handleReturn(int replyCode, String replyText,
String exchange, String routingKey,
AMQP.BasicProperties properties, byte[] body)
throws IOException {
String msg = new String(body);
System.out.println("replyText:"+replyText);
System.out.println("exchange:"+exchange);
System.out.println("routingKey:"+routingKey);
System.out.println("msg:"+msg);
}
}); //声明一组路由键
String[] routingKeys={"error","info","warning"};
//发送消息到交换器上
for(int i=0;i<3;i++){
String routingKey = routingKeys[i%3];
// 发送的消息
String message = "Hello World_"+(i+1)+("_"+System.currentTimeMillis()); //通过路由键把消息发送到交换器上
//参数mandatory: mandatory=true,投递消息时无法找到一个合适的队列,把消息返回给生产者,
// mandatory=false 丢弃消息(缺省)
channel.basicPublish(EXCHANGE_NAME, routingKey, false,
null, message.getBytes());
System.out.println("----------------------------------------------------");
System.out.println(" Sent Message: [" + routingKey +"]:'"+ message + "'");
//sleep一下让程序不快速结束 可以看到RabbitMQ服务器的响应
Thread.sleep(1000);
} // 关闭信道和连接
channel.close();
connection.close();
} }

3. 在OriginalRabbitMQConsumer工程新建一个发送方确认消费者

package study.demo.producerconfirm;

import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException; /**
*
* @Description: 发送方确认消费者
* @author leeSmall
* @date 2018年9月16日
*
*/
public class ProducerConfirmConsumer { private static final String EXCHANGE_NAME = "producer_confirm"; public static void main(String[] argv) throws IOException, TimeoutException {
//1.创建一个连接工厂
ConnectionFactory factory = new ConnectionFactory();
//设置要连接的RabbitMQ服务器的地址
factory.setHost("127.0.0.1"); //2.通过连接工厂创建一个连接
Connection connection = factory.newConnection(); //3.通过连接创建一个信道 信道是用来传送数据的
Channel channel = connection.createChannel(); //4.通过信道声明一个交换器 第一个参数时交换器的名字 第二个参数时交换器的种类
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT); //5.声明队列
String queueName = "producer_confirm";
channel.queueDeclare(queueName,false,false,
false,null); //6.声明一个只消费错误日志的路由键error
String routingKey = "error"; //7.队列通过路由键绑定到交换器上
channel.queueBind(queueName,EXCHANGE_NAME,routingKey);
System.out.println("Waiting message......."); // 8.创建队列消费者 设置一个监听器监听消费消息
final Consumer consumerB = 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");
System.out.println( "Received ["+ envelope.getRoutingKey() + "] "+message);
}
};
//9.消费者自动确认:autoAck参数为true
channel.basicConsume(queueName, true, consumerB);
} }

4. 启动发送方确认消费者ProducerConfirmConsumer,再分别启动生产者(发送方)确认同步模式的类ProducerConfirm和生产者(发送方)确认异步模式ProducerConfirmAsync

启动发送方确认消费者ProducerConfirmConsumer:

启动生产者(发送方)确认同步模式的类ProducerConfirm:

查看发送方确认消费者ProducerConfirmConsumer的状态:

启动生产者(发送方)确认异步模式ProducerConfirmAsync:

查看发送方确认消费者ProducerConfirmConsumer的状态:

示例代码获取地址

消息中间件系列三:使用RabbitMq原生Java客户端进行消息通信(消费者(接收方)自动确认模式、消费者(接收方)自行确认模式、生产者(发送方)确认模式)的更多相关文章

  1. 原生 Java 客户端进行消息通信

    原生 Java 客户端进行消息通信 Direct 交换器 DirectProducer:direct类型交换器的生产者 NormalConsumer:普通的消费者 MulitBindConsumer: ...

  2. RabbitMQ(3) Java客户端使用

    RabbitMQ针对不同的开发语言(java,python,c/++,Go等等),提供了丰富对客户端,方便使用.就Java而言,可供使用的客户端有RabbitMQ Java client. Rabbi ...

  3. 消息中间件系列五:RabbitMQ的使用场景(异步处理、应用解耦)

    一.异步处理 场景: 用户注册,写入数据库成功以后,发送邮件和短信. 准备工作: 1)安装RabbitMQ,参考前面的文章 2)新建一个名为RabbitMQAsyncProc的maven web工程, ...

  4. 消息中间件系列二:RabbitMQ入门(基本概念、RabbitMQ的安装和运行)

    一.基本概念 1. AMQP AMQP,即Advanced Message Queuing Protocol,一个提供统一消息服务的应用层标准高级消息队列协议.支持不同语言和不同的产品 2. 生产者 ...

  5. 喵星之旅-狂奔的兔子-rabbitmq的java客户端使用入门

    一.简介 RabbitMQ是实现了高级消息队列协议(AMQP)的开源消息代理软件(亦称面向消息的中间件). 消息队列都涉及的生产者消费者模型,不做详解,本文只作为快速使用的参考文档. 消息队列主要有点 ...

  6. Hive学习(三)Hive的Java客户端操作

    Hive的Java客户端操作分为JDBC和Thrifit Client,首先启动Hive远程服务: hive --service hiveserver 一.JDBC 在MyEclipse中首先创建连接 ...

  7. Redis系列(三)-Redis发布订阅及客户端编程

    阅读目录 发布订阅模型 Redis中的发布订阅 客户端编程示例 0.3版本Hredis 发布订阅模型 在应用级其作用是为了减少依赖关系,通常也叫观察者模式.主要是把耦合点单独抽离出来作为第三方,隔离易 ...

  8. RabbitMQ学习笔记2-理解消息通信

    消息包含两部分:1.有效载荷(payload) - 你想要传输的数据.2.标签(lable) - 描述有效载荷的相关信息,包含具体的交换器.消息的接受兴趣方等. rabbitmq的基础流程如下: Ra ...

  9. 消息中间件系列四:RabbitMQ与Spring集成

    一.RabbitMQ与Spring集成  准备工作: 分别新建名为RabbitMQSpringProducer和RabbitMQSpringConsumer的maven web工程 在pom.xml文 ...

随机推荐

  1. phpmyadmin#1045 无法登录 MySQL 服务器

    使用lnmp安装wordpress在登录phpmyadmin数据库的时候诡异的出现了   phpmyadmin#1045 无法登录 MySQL 服务器,解决方法如下: 1 修改root的密码—不可以行 ...

  2. 第一章 初始STM32

    1.3什么是STM32? ST是意法半导体,M是MIcroelectronisc的缩写,32表示32位. 合起来就是:ST公司开发的32位微控制器. 1.4 STM32 能做什么? STM32属于一个 ...

  3. BZOJ2240 : ural1676 Mortal Combat

    首先如果最大匹配不足$n$个那么显然每条边都不可能在匹配为$n$的方案中. 对于一条边$(u,v)$,如果它可能在最大匹配中,有两种情况: $1.(u,v)$是当前方案的匹配边. $2.$可以沿着$( ...

  4. 用shell脚本守护后台进程

    假如现在在 crond 中添加了一个每分钟执行的定时任务如下: */ * * * * root cd /data/sbin; sh test.sh >/dev/>& 为了防止上一个 ...

  5. yii2 Gridview网格小部件

    Gridview 网格小部件 一.特点: 1.是yii中功能最强大的小部件之一: 2.非常适合快速建立系统的管理后台. 3.用 dataProvider 键来指定数据的提供者 4.用 filterMo ...

  6. Java中的public、private、protected,函数修饰符

    1.public:public表明该数据成员.成员函数是对所有用户开放的,项目中其他脚本都可以直接进行调用 2.private:private表示私有,私有的意思就是除了脚本之外,项目中其他类都不可以 ...

  7. Delphi 获取当前鼠标下的控件内容

    Delphi 获取当前鼠标下的控件内容 主要函数: GetCursorPos://获取鼠标的位置 WindowFromPoint://获取制定point下的handle GetClassName:// ...

  8. 咏南新CS三层开发框架

    咏南新CS三层开发框架 咏南WEB桌面框架演示:47.106.93.126:9999 咏南WEB手机框架本地:47.106.93.126:8077 咏南CS框架下载:https://pan.baidu ...

  9. 纯 CSS 实现高度与宽度成比例的效果

    http://zihua.li/2013/12/keep-height-relevant-to-width-using-css/

  10. 对于eclipse building workspaces 慢的问题,解决方法

    在项目根目录中有个.project文件,将其中的: <buildCommand> <name>org.eclipse.wst.jsdt.core.javascriptValid ...