🔰 学习视频 🔰
集数:44—47
🔰 学习格言 🔰
练拳不练功,到老一场空;基础不牢,地动山摇。
一、传输协议简介
传输协议介绍:https://activemq.apache.org/configuring-version-5-transports.html
ActiveMQ支持的client-broker通讯协议有: TCP、 NIO、 UDP、 SSL、Http(s)、 VM。
🔶 配置文件
配置Transport Connector的文件在activeMQ安装目录的conf/activemq.xml
中的<transportConnectors>
标签之内。
在上文给出的配置信息中,URI描述信息的头部都的是采用协议名称,例如:
描述amqp协议的监听端口时,采用的URI描述格式为amqp:/....
;
描述Stomp协议的监听端口时,采用的URI描述格式为stomp//....
.。
唯独在进行openwire协议描述时,URI头采用的tcp://...
。这是因为ActiveMQ中默认的消息协议就是openwire。
🔶 控制台界面
在MQ控制台界面可查看到:
🔶 修改配置
在修改之前,先备份
cp activemq.xml activemq.xml.bk
打开activemq.xml
vim activemq.xml
显示行号
:set nu
再根据需求进行修改。
二、传输协议种类
Transport Connector配置示例,参考官网: https://activemq.apache.org/configuring-version-5-transports.html
2.1 TCP
Transmission Control Protocol(TCP)
- 这是默认的Broker配置,TCP的Client监听端口61616</font color=red>
- 在网络传输数据前,必须要序列化数据,消息是通过一个叫wire protocol的来序列化成字节流。默认情况下ActiveMQ把wire protocol叫做OpenWire,它的目的是促使网络上的效率和数据快速交互。
- TCP连接的URI形式如: tcp://hostname:port?key=value&key=value, 后面的参数是可选
- TCP传输的优点:
- TCP协议传输可靠性高,稳定性强
- 高效性:字节流方式传递,效率很高
- 有效性、可用性:应用广泛,支持任何平台
2.2 NIO
New I/O API Protocol(NlO)
- NIO协议和TCP协议类似但NIO更侧重于底层的访问操作。它允许开发人员对同一资源可有更多的client调用和服务端有更多的负载。
- 适合使用NIO协议的场景:
- 可能有大量的Client去连接到Broker上,一般情况下, 大量的Client去连接Broker是被操作系统的线程所限制的。因此,
- NIO的实现比TCP需要更少的线程去运行,所以建议使用NIO协议
- 可能对于Broker有一个很迟钝的网络传输,NIO比TCP提供更好的性能。
- NIO连接的URI形式:
nio//hostname:port?key=value
其他不常用的协议就不再记录了。
如果您不特别指定ActiveMQ的网络监听端口,那么这些端口都将使用BIO网络IO模型。( OpenWire, STOMP, AMQ…)
所以为了首先提高单节点的网络吞吐性能,我们需要明确指定Active的网络IO模型,URI格式头以nio
开头,表示这个端口使用以TCP协议为基础的NIO网络IO模型。
三、NIO协议演示
步骤1:停止activemq运行
cd mq的bin目录下
./activemq stop
步骤2:修改配置协议activemq.conf
新增:<transportConnector name="nio" uri="nio://0.0.0.0:61618?trace=true/>"
步骤3:开放61618端口
设置开放的端口号
firewall-cmd --add-service=http --permanent
sudo firewall-cmd --add-port=61618/tcp --permanent
重启防火墙
firewall-cmd --reload
查看开放端口号
firewall-cmd --list-all
步骤4:启动
cd mq的bin目录下
./activemq start
在MQ控制台界面可查看到:
3.1 测试生产者
import org.apache.activemq.ActiveMQConnectionFactory;
import javax.jms.*;
public class JmsProduce {
public static final String ACTIVEMQ_URL = "nio://192.168.150.101:61618";
public static final String QUEUE_NAME = "queue01";
public static void main(String[] args) throws JMSException {
// 1 创建连接工厂,按照给定的url地址,采用默认的用户名和密码
ActiveMQConnectionFactory activeMQConnectionFactory = new ActiveMQConnectionFactory(ACTIVEMQ_URL);
// 2 通过连接工厂,获得连接connection并启动访问
Connection connection = activeMQConnectionFactory.createConnection();
connection.start();
// 3 创建会话session
// 参数1:事务
// 参数2:签收
Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
// 4 创建目的地(队列还是主题,这里选择队列)
Queue queue = session.createQueue(QUEUE_NAME);
// 5 创建消息的生产者
MessageProducer messageProducer = session.createProducer(queue);
messageProducer.setDeliveryMode(DeliveryMode.PERSISTENT);
// 6 通过使用messageProducer生产3条消息发送到MQ队列中
for (int i = 0; i < 3; i++) {
// 7 创建消息 按照要求写好的消息
TextMessage textMessage = session.createTextMessage("msg----" + i);//一个字符串
textMessage.setStringProperty("c01", "vip");
// 8 通过messageProducer发给mq
messageProducer.send(textMessage);
}
// 9 关闭资源
messageProducer.close();
session.close();
connection.close();
System.out.println("---------消息发布到MQ完成--------");
}
}
消息发布成功。
3.2 测试消费者
import org.apache.activemq.ActiveMQConnectionFactory;
import javax.jms.*;
import java.io.IOException;
public class JmsConsumer {
public static final String ACTIVEMQ_URL = "nio://192.168.150.101:61618";
public static final String QUEUE_NAME = "queue01";
public static void main(String[] args) throws JMSException, IOException {
// 1 创建连接工厂,按照给定的url地址,采用默认的用户名和密码
ActiveMQConnectionFactory activeMQConnectionFactory = new ActiveMQConnectionFactory(ACTIVEMQ_URL);
// 2 通过连接工厂,获得连接connection并启动访问
Connection connection = activeMQConnectionFactory.createConnection();
connection.start();
// 3 创建会话session
// 参数1:事务
// 参数2:签收
Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
// 4 创建目的地(队列还是主题,这里选择队列)
Queue queue = session.createQueue(QUEUE_NAME);
// 5 创建消息的消费者
MessageConsumer messageConsumer = session.createConsumer(queue);
// 6 处理消息
/*
方式1:同步阻塞方式(receive())
订阅者或接收者调用MessageConsumer的receive()方法来接收消息,
receive方法在能够接收到消息之前(或超时之前)将一直堵塞。
receive(xxxx),如果添加参数表示等待一定时间,超过时间将关闭。
*/
// while (true) {
// TextMessage textMessage = (TextMessage) messageConsumer.receive();
// if (textMessage!=null) {
// System.out.println("-------消费者接收到消息: " + textMessage.getText());
// } else {
// break;
// }
// }
/*
方式2: 异步非阻塞方式(监听器onMessage())
订阅者或接收者通过MessageConsumer的setMessageListener(MessageListener listener)注册一个消息监听器,
当消息到达后,系统自动调用监听器MessageListener的onMessage(Message message)方法。
*/
messageConsumer.setMessageListener(new MessageListener() {
@Override
public void onMessage(Message message) {
if (message != null && message instanceof TextMessage) {
TextMessage textMessage = (TextMessage) message;
try {
System.out.println("-------消费者接收到消息: " + textMessage.getText());
} catch (JMSException e) {
e.printStackTrace();
}
}
}
});
System.in.read(); // 保证控制台不灭,消费者启动之后将一直等待。如果关闭的话,消费者启动后还没监听到就立刻关闭了。
// 7 资源关闭
messageConsumer.close();
session.close();
connection.close();
}
}
消息接收成功。
四、NIO案例增强
上述NIO性能不错了,如何进一步优化?
URI格式头以nio
开头,表示这个端口使用以TCP协议为基础的NIo网络I0模型。但是这样的设置方式,只能使这个端口支持Openwire协议。
那么我们怎么既让这个端口支持NIO网络IO模型,又让它支持多个协议呢?
步骤1:停止activemq运行
cd mq的bin目录下
./activemq stop
步骤2:修改配置协议activemq.conf
新增:<transportConnector name="auto+nio" uri="auto+nio://0.0.0.0:61618/>"
步骤3:开放61618端口
同第三部分。
步骤4:启动
cd mq的bin目录下
./activemq start
在MQ控制台界面可查看到:
步骤5:运行生产者、消费者程序
传输协议可写为:
nio://192.168.150.101:61618
也可以换成,即使刚刚的tcp端口为61616
tcp://192.168.150.101:61618