RabbitMQ简单介绍

转载自:http://blog.csdn.net/dandanzmc/article/details/52244529

RabbitMQ在我看来就是AMQP(Advanced Message Queuing Protocol)标准协议的一个实现,至于这个AMQP的标准协议具体是怎么定的,我不太清楚,没有深入的理解。都说RabbitMQ是一个很好的分布式MQ,我想这也是因为RabbitMQ是用Erlang语言开发出来的缘故,Erlang语言是专门为开发concurrent和distribution系统的一种语言,它能很好的对分布式和并发做支持,当然,我不会Erlang,所以具体的原理我也不知道了。

      RabbitMQ主要是有这么几个部分,个人觉得这张图能说明RabbitMQ:

     


Publisher:

          是Message的生产者,Publisher这个Clients产生了一些Message。


Consumer:

          Message的消费者,Publisher产生的Message,最终要到达Consumer这个Clients,进行消费。


Exchange:

         指定消息按什么规则,路由到哪个Queue,Message消息先要到达Exchange,在Server中承担着从Produce接收Message的责任。


Queue:

         到达Exchange的消息,根据制定的规则(Routing key)到达对应的Queue,在Server中承担着装载Message,是Message的容器,等待被消费出去。


Routing key:

         在Exchange和Queue之间隐藏有一条黑线,可以将这条黑线看成是Routing key,Exchange就是根据这些定义好的Routing key将Message送到对应的Queue中去,是Exchange和Queue之间的桥梁。


Broker:

        之前一直不理解这个Broker,其实Broker就是接收和分发消息的应用,也就是说RabbitMQ Server就是Message Broker。


VirtualHost:

        虚拟主机,一个Broker里可以开有多个VirtualHost,它的作用是用作不同用户的权限分离。 


Connection:

       是Publisher/Consumer和Broker之间的TCP连接。断开连接的操作只会在Publisher/Consumer端进行,Broker不会断开连接,除非出现网络故障或者Broker服务出现问题,Broker服务宕了。


Connection: Channel: 

        如果每一次访问RabbitMQ就建立一个Connection,那在消息量大的时候建立TCP Connection的开销就会很大,导致的后果就是效率低下。


左边的Client向右边的Client发送消息,流程:
       第一:获取Conection
       第二:获取Channel
       第三:定义Exchange,Queue
       第四:使用一个RoutingKey将Queue Binding到一个Exchange上
       第五:通过指定一个Exchange和一个RoutingKey来将消息发送到对应的Queue上,
       第六:Consumer在接收时也是获取connection,接着获取channel,然后指定一个Queue,到Queue上取消息,它对Exchange,RoutingKey及如何Binding都不关心,到对应的Queue上去取消息就行了。
       

        一个Publisher Client发送消息,哪些Consumer Client可以收到消息,在于Exchange,RoutingKey,Queue的关系上。


       RabbitMQ的简单理论知识就介绍这么多,下篇博客介绍RabbitMQ在CentoOS7上的安装和配置。


代码实例(Publisher):

/** 
         * 创建连接连接到MabbitMQ 
         */ 
ConnectionFactory factory = new ConnectionFactory();
//设置MabbitMQ所在主机ip或者主机名
factory.setHost("host");
factory.setPort("port");
factory.setUsername("username");
factory.setPassword("password");
factory.setVirtualHost("vhost");
Connection connection = null;
try {
//主题式路由
String EXCHANGE_NAME = "amq.topic";
//队列名称
String queueName = queue.queuereceive;
//创建一个连接
connection = factory.newConnection();
//创建一个频道
Channel channel = connection.createChannel();
//指定一个队列
//queueDeclare第一个参数表示队列名称;第二个参数为是否持久化(true表示是,队列将在服务器重启时生存)、
//第三个参数为是否是独占队列(创建者可以使用的私有队列,断开后自动删除)、
//第四个参数为当所有消费者客户端连接断开时是否自动删除队列、第五个参数为队列的其他参数
channel.queueDeclare(queueName, false, false, false, null);
//往队列中发出一条消息,持久化消息
//第一个参数为交换机名称、第二个参数为队列映射的路由key、第三个参数为消息的其他属性、第四个参数为发送信息的主体
channel.basicPublish(EXCHANGE_NAME, queueName, null,message.getBytes());
//关闭频道 
channel.close();
} catch (IOException | TimeoutException e) {
e.printStackTrace();
} finally {
if (connection != null) {
try {
//关闭连接  
connection.close();
} catch (IOException _ignore) {


}
}
}



代码实例(Consumer):

ConnectionFactory factory = new ConnectionFactory();
factory.setHost("host");
factory.setPort("port");
factory.setUsername("username");
factory.setPassword("password");
factory.setVirtualHost("vhost");
//队列名称
String queueName = queue.queuepush;
//主题式路由
String exchange = "amq.topic";
System.out.println("QUEUE INIT:" + queueName);
try {
Connection connection = factory.newConnection();
Channel channel = connection.createChannel();
//声明队列,主要为了防止消息接收者先运行此程序,队列还不存在时创建队列。  
channel.queueDeclare(queueName, false, false, false, null); 
channel.queueBind(queueName, exchange, queueName);
final Consumer consumer = 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");
try{

    //根据得到的参数进行业务处理
DoSomthing(message)
}catch(Exception ex){
ex.printStackTrace();
}
}
};
//指定消费队列  
channel.basicConsume(queueName, true, consumer);
} catch (IOException e) {
e.printStackTrace();
} catch (TimeoutException e) {
e.printStackTrace();
}




评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值