https://chrome.google.com/webstore/detail/markdown-viewer/ckkdlimhmcjmikdlpkmbgfkaikojcbjk 访问将插件添加到google浏览器
直接在chrome浏览器的网址栏输入chrome://extensions/ 查看已经下载的扩展程序,然后找到Markdown Viewer,点击详细信息,然后将允许访问文件网址勾选。这样以后就可以把想要查看的Markdown文件直接拖进浏览器里就可以查看编译过后好看的排版。
把.md文件在google浏览器上访问,或者在vscode上安装.md插件预览即可
推荐使用Typora软件,下载安装即可,方便书写
2. 安装完成后,输入以下命令
rabbitmq-plugins enable rabbitmq_management
再在google浏览器上输入 localhost:15672,如果无响应,在自己的C:\Users\(用户名)\AppData\Roaming\RabbitMQ\db路径下的文件全部删除,再重新下载rabbitmq-server,再重复2.即可
3. 看到以下窗口即成功,账户密码皆为guest,此处涉及对rabbitmq的理解,可以理解为mysql数据库。
4. virtual hosts<->mysql中的数据库,一般以/开头,然后对用户进行授权,右边栏中有virtual hosts,进行授权,授权完毕,就可以用自己的账户密码了
1. 此处是pom.xml的依赖,用maven自动导入,依赖在maven官网可以找到,一定要找groupId和artifactId相同的,且使用非红名测试版的最新版
<dependency>
<groupId>com.rabbitmqgroupId>
<artifactId>amqp-clientartifactId>
<version>5.7.3version>
dependency>
<dependency>
<groupId>org.slf4jgroupId>
<artifactId>slf4j-apiartifactId>
<version>1.7.26version>
dependency>
<dependency>
<groupId>org.slf4jgroupId>
<artifactId>slf4j-log4j12artifactId>
<version>1.7.26version>
<scope>testscope>
dependency>
<dependency>
<groupId>log4jgroupId>
<artifactId>log4jartifactId>
<version>1.2.17version>
dependency>
<dependency>
<groupId>junitgroupId>
<artifactId>junitartifactId>
<version>4.12version>
<scope>testscope>
dependency>
2. 获取MQ连接
写一个工具类util,像tp5里的database.php一样,这样就可以调用了
package com.myapp.demoesi.util;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
/*包是自动导入的,不用考虑那么多*/
public class ConnectionUtils {
/*获取MQ的连接,此连接只调用在局部static,并且返回值是一个MQ连接*/
public static Connection getConnection() throws IOException, TimeoutException {
/*定义一个连接工厂,相当于mysql中的$conn = mysqli_connect($servername, $username, $password);*/
ConnectionFactory factory = new ConnectionFactory();
//设置服务地址$servername
factory.setHost("127.0.0.1");
//AMQP 5672(http)
factory.setPort(5672);
//vhost(mysql中的数据库)
factory.setVirtualHost("/vhost_mmr");
//用户名$username
factory.setUsername("mmr");
//密码$password
factory.setPassword("123456");
return factory.newConnection();
}
}
3.1 simple简单队列
P:发送;红色:队列;绿色:消费
新建一个simple包,在里面创建Send和Resv类
3.1.1 Send:
package com.myapp.demoesi.rabbitmq.simple;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Send {
private static final String QUEUE_NAME = "test_simple_queue";
//私有,静态,不可变
public static void main(String[] args) throws IOException, TimeoutException {
//获取一个链接,在刚刚的工具类里
Connection connection = ConnectionUtils.getConnection();
//从连接中获取一个通道
Channel channel = connection.createChannel();
//创建队列声明
channel.queueDeclare(QUEUE_NAME, false, false, false, null);
String msg = "Hello,Simple!";
//发送在这个队列
channel.basicPublish("", QUEUE_NAME, null, msg.getBytes());
System.out.println("--send msg:" + msg);
//信道和连接关了
channel.close();
connection.close();
}
}
运行完毕进行验证,在你的rabbitmq里的Queues,进入你的/vhost_mmr的test_simple_queue里
观察是否可以接收到发送的消息,如果可以Send就成功啦!
3.1.2 Resv:
package com.myapp.demoesi.rabbitmq.simple;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Resv {
private static final String QUEUE_NAME = "test_simple_queue";
public static void main(String[] args) throws IOException, TimeoutException {
//获取链接
Connection connection = ConnectionUtils.getConnection();
//创建频道
Channel channel = connection.createChannel();
//队列声明,保险起见
channel.queueDeclare(QUEUE_NAME, false, false, false, null);
DefaultConsumer consumer = new DefaultConsumer(channel) {
//获取到达的消息
@Override
public void handleDelivery(String consumerTag, Envelope envelope,
AMQP.BasicProperties properties, byte[] body) throws IOException {
super.handleDelivery(consumerTag, envelope, properties, body);
String msg = new String(body, "UTF-8");
System.out.println("new api resv: " + msg);
}
};
//监听队列 android
channel.basicConsume(QUEUE_NAME, true, consumer);
}
/*下面是老方法,此处供学习参考,redis也差不多这样,这个网址聊天这个挺不错的https://www.imooc.com/learn/758*/
public void oldresv() {
/* //获取链接
Connection connection = ConnectionUtils.getConnection();
//创建频道
Channel channel = connection.createChannel();
//定义队列的消费者
QueueingConsumer consumer = new QueueingConsumer(channel);
//监听队列
channel.basicConsume(QUEUE_NAME,true,consumer);
while(true){
Delivery delivery = consumer.nextDelivery();
String msgString = new String(delivery.getBody());
System.out.println("[resv]msg:"+msgString);
}*///老方法1.3
}
}
先开Resv,再执行Send,就可以看到我们发送的Hello simple!了。
3.2.1 work工作队列(轮训分发)
3.2.1.1 Send: 在队列里攒了50个消息
package com.myapp.demoesi.rabbitmq.work;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Send {
/*
*p---queue
*/
private static final String QUEUE_NAME = "test_work_queue";
public static void main(String[] args) throws IOException, TimeoutException, InterruptedException {
//获取连接
Connection connection = ConnectionUtils.getConnection();
//获取channel
Channel channel = connection.createChannel();
//声明队列
channel.queueDeclare(QUEUE_NAME,false,false,false,null);
for (int i = 0;i<50;i++){
String msg = "Hello,"+i;
System.out.println("[work] send: "+msg);
channel.basicPublish("",QUEUE_NAME,null,msg.getBytes());
Thread.sleep(i*20);
}
channel.close();
connection.close();
}
}
3.2.1.2 Resv1:消费者1
package com.myapp.demoesi.rabbitmq.work;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Resv1 {
private static final String QUEUE_NAME = "test_work_queue";
public static void main(String[] args) throws IOException, TimeoutException {
//获取连接
Connection connection = ConnectionUtils.getConnection();
//获取channel
Channel channel = connection.createChannel();
//S声明队列
channel.queueDeclare(QUEUE_NAME, false, false, false, null);
//定义一个消费者
Consumer consumer = new DefaultConsumer(channel) {
//消息到达触发这个方法
@Override
public void handleDelivery(String consumerTag, Envelope envelope,
AMQP.BasicProperties properties, byte[] body) throws IOException {
super.handleDelivery(consumerTag, envelope, properties, body);
String msg = new String(body, "UTF-8");
System.out.println("[resv1] msg:" + msg);
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
System.out.println("[1] done");
}
}
};
boolean autoAck = true;
channel.basicConsume(QUEUE_NAME,autoAck,consumer);
}
}
3.2.2.3 Resv2:消费者2
package com.myapp.demoesi.rabbitmq.work;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Resv2 {
private static final String QUEUE_NAME = "test_work_queue";
public static void main(String[] args) throws IOException, TimeoutException {
//获取连接
Connection connection = ConnectionUtils.getConnection();
//获取channel
Channel channel = connection.createChannel();
//S声明队列
channel.queueDeclare(QUEUE_NAME, false, false, false, null);
//定义一个消费者
Consumer consumer = new DefaultConsumer(channel) {
//消息到达触发这个方法
@Override
public void handleDelivery(String consumerTag, Envelope envelope,
AMQP.BasicProperties properties, byte[] body) throws IOException {
super.handleDelivery(consumerTag, envelope, properties, body);
String msg = new String(body, "UTF-8");
System.out.println("[resv1] msg:" + msg);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
System.out.println("[2] done");
}
}
};
boolean autoAck = true;
channel.basicConsume(QUEUE_NAME,autoAck,consumer);
}
}
3.2.2 workfair工作队列(公平分发)
3.2.2.1 Send: 在队列里攒了50个消息
package com.myapp.demoesi.rabbitmq.workfair;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.MessageProperties;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Send {
/*
*p---queue
*/
private static final String QUEUE_NAME = "test_workfair_queue";
public static void main(String[] args) throws IOException, TimeoutException, InterruptedException {
//获取连接
Connection connection = ConnectionUtils.getConnection();
//获取channel
Channel channel = connection.createChannel();
//声明队列
boolean durable = true;
channel.queueDeclare(QUEUE_NAME, durable, false, false, null);/*消息持久化
每个消费者发送确认消息之前,消息队列不发送下一个消息到消费者,一次只处理一个消息
durable持久化,但在声明队列后不能更改参数,因为队列已经存在;但消息不一定持久化*/
int prefetchCount = 1;
channel.basicQos(prefetchCount);//限制发送给同一个消费者不得超过一个消息
for (int i = 0; i < 50; i++) {
String msg = "Hello," + i;
System.out.println("[work] send: " + msg);
channel.basicPublish("", QUEUE_NAME, MessageProperties.PERSISTENT_TEXT_PLAIN, msg.getBytes());
Thread.sleep(i * 5);
}
channel.close();
connection.close();
}
}
3.2.2.2 Resv1:消费者1
package com.myapp.demoesi.rabbitmq.workfair;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Resv1 {
private static final String QUEUE_NAME = "test_workfair_queue";
public static void main(String[] args) throws IOException, TimeoutException {
//获取连接
Connection connection = ConnectionUtils.getConnection();
//获取channel
Channel channel = connection.createChannel();
//S声明队列
boolean durable = true;
channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化
channel.basicQos(1);
//定义一个消费者
Consumer consumer = new DefaultConsumer(channel) {
int ing = 1;
//消息到达触发这个方法
@Override
public void handleDelivery(String consumerTag, Envelope envelope,
AMQP.BasicProperties properties, byte[] body) throws IOException {
super.handleDelivery(consumerTag, envelope, properties, body);
String msg = new String(body, "UTF-8");
System.out.println("[resv1] msg:" + msg);
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
System.out.println("[" + ing + "] done");
ing++;
channel.basicAck(envelope.getDeliveryTag(), false);
}
}
};
boolean autoAck = false;//自动应答
channel.basicConsume(QUEUE_NAME, autoAck, consumer);
}
}
3.2.2.3 Resv2:消费者2
package com.myapp.demoesi.rabbitmq.workfair;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Resv2 {
private static final String QUEUE_NAME = "test_workfair_queue";
public static void main(String[] args) throws IOException, TimeoutException {
//获取连接
Connection connection = ConnectionUtils.getConnection();
//获取channel
Channel channel = connection.createChannel();
//S声明队列
boolean durable = true;
channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化
channel.basicQos(1);
//定义一个消费者
Consumer consumer = new DefaultConsumer(channel) {
int ing = 1;
//消息到达触发这个方法
@Override
public void handleDelivery(String consumerTag, Envelope envelope,
AMQP.BasicProperties properties, byte[] body) throws IOException {
super.handleDelivery(consumerTag, envelope, properties, body);
String msg = new String(body, "UTF-8");
System.out.println("[resv1] msg:" + msg);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
System.out.println("[" + ing + "] done");
ing++;
//手动回执一个消息
channel.basicAck(envelope.getDeliveryTag(), false);
}
}
};
boolean autoAck = false;
channel.basicConsume(QUEUE_NAME, autoAck, consumer);//自动应答
}
}
3.3 publish_subscribe订阅模式 fanout(不处理路由键)
3.3.1 Send: 发送到交换机
package com.myapp.demoesi.rabbitmq.ps;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.MessageProperties;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Send {
private static final String EXCHANGE_NAME="test_exchange_fanout";
public static void main(String[] args) throws IOException, TimeoutException {
Connection connection = ConnectionUtils.getConnection();
Channel channel=connection.createChannel();
//声明交换机
channel.exchangeDeclare(EXCHANGE_NAME,"fanout");/*分发
交换机没有存储消息的功能,队列才有
一方面接收生产者的消息,另一方面是向对列推送消息
匿名转发:""
fanout(不处理路由键)
direct(处理路由键)
*/
//发送消息
String msg="Hello ps !";
channel.basicPublish(EXCHANGE_NAME,"" , MessageProperties.PERSISTENT_TEXT_PLAIN, msg.getBytes());
System.out.println("Send: "+msg);
channel.close();
connection.close();
}
}
3.3.2 Resv1: 交换机绑定队列
package com.myapp.demoesi.rabbitmq.ps;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Resv1 {
private static final String QUEUE_NAME = "test_queue_fanout_email";
private static final String EXCHANGE_NAME = "test_exchange_fanout";
public static void main(String[] args) throws IOException, TimeoutException {
//获取连接
Connection connection = ConnectionUtils.getConnection();
//获取channel
Channel channel = connection.createChannel();
//S声明队列
boolean durable = true;
channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化
//绑定队列到交换机转发器
channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, "");
channel.basicQos(1);
//定义一个消费者
System.out.println("1 666");
Consumer consumer = new DefaultConsumer(channel) {
int ing = 1;
//消息到达触发这个方法
@Override
public void handleDelivery(String consumerTag, Envelope envelope,
AMQP.BasicProperties properties, byte[] body) throws IOException {
super.handleDelivery(consumerTag, envelope, properties, body);
String msg = new String(body, "UTF-8");
System.out.println("[resv1] msg:" + msg);
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
System.out.println("[" + ing + "] done");
ing++;
channel.basicAck(envelope.getDeliveryTag(), false);
}
}
};
boolean autoAck = false;//自动应答
channel.basicConsume(QUEUE_NAME, autoAck, consumer);
}
}
3.3.3 Resv2: 交换机绑定队列
package com.myapp.demoesi.rabbitmq.ps;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Resv2 {
private static final String QUEUE_NAME = "test_queue_fanout_sms";
private static final String EXCHANGE_NAME = "test_exchange_fanout";
public static void main(String[] args) throws IOException, TimeoutException {
//获取连接
Connection connection = ConnectionUtils.getConnection();
//获取channel
Channel channel = connection.createChannel();
//S声明队列
boolean durable = true;
channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化
//绑定队列到交换机转发器
channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, "");
channel.basicQos(1);
//定义一个消费者
System.out.println("2 666");
Consumer consumer = new DefaultConsumer(channel) {
int ing = 1;
//消息到达触发这个方法
@Override
public void handleDelivery(String consumerTag, Envelope envelope,
AMQP.BasicProperties properties, byte[] body) throws IOException {
super.handleDelivery(consumerTag, envelope, properties, body);
String msg = new String(body, "UTF-8");
System.out.println("[resv2] msg:" + msg);
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
System.out.println("[" + ing + "] done");
ing++;
channel.basicAck(envelope.getDeliveryTag(), false);
}
}
};
boolean autoAck = false;//自动应答
channel.basicConsume(QUEUE_NAME, autoAck, consumer);
}
}
3.4 路由模式 direct(处理路由键)
3.4.1 Send: 发送携带路由键的消息给交换机
package com.myapp.demoesi.rabbitmq.routing;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.MessageProperties;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Send {
private static final String EXCHANGE_NAME="test_exchange_direct";
public static void main(String[] args) throws IOException, TimeoutException {
Connection connection = ConnectionUtils.getConnection();
Channel channel = connection.createChannel();
//exchange
channel.exchangeDeclare(EXCHANGE_NAME,"direct");
String routingKey = "info";
String msg = "Hello, "+routingKey+"! ";
channel.basicPublish(EXCHANGE_NAME, msg, MessageProperties.PERSISTENT_TEXT_PLAIN, routingKey.getBytes());
System.out.println("Send: "+msg);
channel.close();
connection.close();
}
}
3.4.2 Resv1: 绑定"error"路由键和队列"test_queue_direct_1"到交换机上
package com.myapp.demoesi.rabbitmq.routing;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Resv1 {
private static final String EXCHANGE_NAME = "test_exchange_direct";
private static final String QUEUE_NAME = "test_queue_direct_1";
public static void main(String[] args) throws IOException, TimeoutException {
//获取连接
Connection connection = ConnectionUtils.getConnection();
//获取channel
Channel channel = connection.createChannel();
boolean durable = true;
channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化
channel.queueBind(QUEUE_NAME,EXCHANGE_NAME,"error");
channel.basicQos(1);
Consumer consumer = new DefaultConsumer(channel) {
int ing = 1;
//消息到达触发这个方法
@Override
public void handleDelivery(String consumerTag, Envelope envelope,
AMQP.BasicProperties properties, byte[] body) throws IOException {
super.handleDelivery(consumerTag, envelope, properties, body);
String msg = new String(body, "UTF-8");
System.out.println("[resv1] msg:" + msg);
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
System.out.println("[" + ing + "] done");
ing++;
channel.basicAck(envelope.getDeliveryTag(), false);
}
}
};
boolean autoAck = false;//自动应答
channel.basicConsume(QUEUE_NAME, autoAck, consumer);
}
}
3.4.3 Resv2: 绑定"warning"“info”"error"路由键和队列"test_queue_direct_2"到交换机上
package com.myapp.demoesi.rabbitmq.routing;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Resv2 {
private static final String EXCHANGE_NAME = "test_exchange_direct";
private static final String QUEUE_NAME = "test_queue_direct_2";
public static void main(String[] args) throws IOException, TimeoutException {
//获取连接
Connection connection = ConnectionUtils.getConnection();
//获取channel
Channel channel = connection.createChannel();
boolean durable = true;
channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化
channel.queueBind(QUEUE_NAME,EXCHANGE_NAME,"error");
channel.queueBind(QUEUE_NAME,EXCHANGE_NAME,"info");
channel.queueBind(QUEUE_NAME,EXCHANGE_NAME,"warning");
channel.basicQos(1);
Consumer consumer = new DefaultConsumer(channel) {
int ing = 1;
//消息到达触发这个方法
@Override
public void handleDelivery(String consumerTag, Envelope envelope,
AMQP.BasicProperties properties, byte[] body) throws IOException {
super.handleDelivery(consumerTag, envelope, properties, body);
String msg = new String(body, "UTF-8");
System.out.println("[resv2] msg:" + msg);
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
System.out.println("[" + ing + "] done");
ing++;
channel.basicAck(envelope.getDeliveryTag(), false);
}
}
};
boolean autoAck = false;//自动应答
channel.basicConsume(QUEUE_NAME, autoAck, consumer);
}
}
3.5 topic主题模式 (通配符处理*、#)
3.5.1 Send: routingKey是"goods.delete",可以让goods.#和#.delete绑定的队列接收到
package com.myapp.demoesi.rabbitmq.topic;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.MessageProperties;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Send {
private static final String EXCHANGE_NAME = "test_exchange_topic";
public static void main(String[] args) throws IOException, TimeoutException {
Connection connection = ConnectionUtils.getConnection();
Channel channel = connection.createChannel();
//exchange
channel.exchangeDeclare(EXCHANGE_NAME, "topic");
String routingKey = "goods.delete";
String msg = "商品.... " + routingKey + "! ";
channel.basicPublish(EXCHANGE_NAME, routingKey, MessageProperties.PERSISTENT_TEXT_PLAIN, routingKey.getBytes());
System.out.println("Send: " + msg);
channel.close();
connection.close();
}
}
3.5.2 Resv1: 绑定goods.add的"test_queue_topic_1"接收goods.add消息
package com.myapp.demoesi.rabbitmq.topic;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Resv1 {
private static final String EXCHANGE_NAME = "test_exchange_topic";
private static final String QUEUE_NAME = "test_queue_topic_1";
public static void main(String[] args) throws IOException, TimeoutException {
//获取连接
Connection connection = ConnectionUtils.getConnection();
//获取channel
Channel channel = connection.createChannel();
boolean durable = true;
channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化
channel.queueBind(QUEUE_NAME,EXCHANGE_NAME,"goods.add");
channel.basicQos(1);
Consumer consumer = new DefaultConsumer(channel) {
int ing = 1;
//消息到达触发这个方法
@Override
public void handleDelivery(String consumerTag, Envelope envelope,
AMQP.BasicProperties properties, byte[] body) throws IOException {
super.handleDelivery(consumerTag, envelope, properties, body);
String msg = new String(body, "UTF-8");
System.out.println("[resv1] msg:" + msg);
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
System.out.println("[" + ing + "] done");
ing++;
channel.basicAck(envelope.getDeliveryTag(), false);
}
}
};
boolean autoAck = false;//自动应答
channel.basicConsume(QUEUE_NAME, autoAck, consumer);
}
}
3.5.3 Resv2: 绑定goods.#的"test_queue_topic_2"接收goods.#的消息
package com.myapp.demoesi.rabbitmq.topic;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Resv2 {
private static final String EXCHANGE_NAME = "test_exchange_topic";
private static final String QUEUE_NAME = "test_queue_topic_2";
public static void main(String[] args) throws IOException, TimeoutException {
//获取连接
Connection connection = ConnectionUtils.getConnection();
//获取channel
Channel channel = connection.createChannel();
boolean durable = true;
channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化
channel.queueBind(QUEUE_NAME,EXCHANGE_NAME,"goods.#");
channel.basicQos(1);
Consumer consumer = new DefaultConsumer(channel) {
int ing = 1;
//消息到达触发这个方法
@Override
public void handleDelivery(String consumerTag, Envelope envelope,
AMQP.BasicProperties properties, byte[] body) throws IOException {
super.handleDelivery(consumerTag, envelope, properties, body);
String msg = new String(body, "UTF-8");
System.out.println("[resv2] msg:" + msg);
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
System.out.println("[" + ing + "] done");
ing++;
channel.basicAck(envelope.getDeliveryTag(), false);
}
}
};
boolean autoAck = false;//自动应答
channel.basicConsume(QUEUE_NAME, autoAck, consumer);
}
}
3.1 在rabbitmq中 我们可以通过持久化数据 解决rabbitmq服务器异常 的数据丢失问题
问题:生产者将消息发送出去之后,消息到底有没有到达 rabbitmq 服务器,默认的情况是不知道的;
两种方式:
AMQP实现了事务机制,
Confirm模式
3.2 事务机制
txSelect txCommit txRollback
txSelect: 用户将当前channel设置成transation模式
txCommit: 用于提交事务
txRollback: 回滚事务
3.2.1 TxSend:发送translation模式的事务
package com.myapp.demoesi.rabbitmq.tx;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class TxSend {
private static final String QUEUE_NAME = "test_queue_tx";
public static void main(String[] args) throws IOException, TimeoutException {
Connection connection = ConnectionUtils.getConnection();
Channel channel = connection.createChannel();
channel.queueDeclare(QUEUE_NAME,false,false,false,null);
String msg="Hello tx massage! ";
try {
channel.txSelect();
channel.basicPublish("",QUEUE_NAME,null,msg.getBytes());
//int xx = 1/0;
channel.txCommit();
}catch (Exception e){
channel.txRollback();
System.out.println(" send message rollback");
}
channel.close();
connection.close();
}
}
3.2.2 TxResv:正常接收
package com.myapp.demoesi.rabbitmq.tx;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class txResv {
private static final String QUEUE_NAME="test_queue_tx";
public static void main(String[] args) throws IOException, TimeoutException {
Connection connection = ConnectionUtils.getConnection();
Channel channel = connection.createChannel();
channel.queueDeclare(QUEUE_NAME,false,false,false,null);
channel.basicConsume(QUEUE_NAME,true,
new DefaultConsumer(channel){
@Override
public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
super.handleDelivery(consumerTag, envelope, properties, body);
System.out.println("resv[tx]"+new String(body,"utf-8"));
}
});
}
}
此种模式还是很耗时的,采用这种方式,降低了rabbitmq的吞吐量
3.3 Confirm模式
Confirm模式最大的好处在于他是异步
Nack消息
开启confirm模式
channel.confirmSelect()
编程模式:
1.普通 发一条 waitForConfirms()
2.批量的 发一批 waitForConfirms()
3.异步confirm模式:提供一个回调(我并不会)
3.3.1 Confirm单条: Send1
package com.myapp.demoesi.rabbitmq.confirm;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
/*普通模式*/
public class Send1 {
private static final String QUEUE_NAME = "test_queue_confirm1";
public static void main(String[] args) throws IOException, TimeoutException, InterruptedException {
Connection connection = ConnectionUtils.getConnection();
Channel channel = connection.createChannel();
channel.queueDeclare(QUEUE_NAME,false,false,false,null);
//生产者调用confirmSelect将channel设置为confirm模式 注意
channel.confirmSelect();
String msg="Hello confirm1 massage! ";
channel.basicPublish("",QUEUE_NAME,null,msg.getBytes());
//int xx = 1/0;
if (!channel.waitForConfirms()){
System.out.println("massage send failed");
}else{
System.out.println("massage send ok");
}
channel.close();
connection.close();
}
}
3.3.2 Confirm批量: Send2
package com.myapp.demoesi.rabbitmq.confirm;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
/*普通模式*/
public class Send2 {
private static final String QUEUE_NAME = "test_queue_confirm1";
public static void main(String[] args) throws IOException, TimeoutException, InterruptedException {
Connection connection = ConnectionUtils.getConnection();
Channel channel = connection.createChannel();
channel.queueDeclare(QUEUE_NAME, false, false, false, null);
//生产者调用confirmSelect将channel设置为confirm模式 注意
channel.confirmSelect();
String msg = "Hello confirm2 massage! ";
for (int i = 0; i < 10; i++) {
channel.basicPublish("", QUEUE_NAME, null, msg.getBytes());
}
//int xx = 1/0;
if (!channel.waitForConfirms()) {
System.out.println("massage send failed");
} else {
System.out.println("massage send ok");
}
channel.close();
connection.close();
}
}
3.3.3 Resv:
package com.myapp.demoesi.rabbitmq.confirm;
import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Resv {
private static final String QUEUE_NAME="test_queue_confirm1";
public static void main(String[] args) throws IOException, TimeoutException {
Connection connection = ConnectionUtils.getConnection();
Channel channel = connection.createChannel();
channel.queueDeclare(QUEUE_NAME,false,false,false,null);
channel.basicConsume(QUEUE_NAME,true,
new DefaultConsumer(channel){
@Override
public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
super.handleDelivery(consumerTag, envelope, properties, body);
System.out.println("resv[confirm]"+new String(body,"utf-8"));
}
});
}
}
很简单吧?
你学会了吗?