java rabbitmq_rabbitMQ第二篇:java简单的实现RabbitMQ

前言:在这里我将用java来简单的实现rabbitMQ。下面我们带着下面问题来一步步的了解和学习rabbitMQ。

1:如果消费者连接中断,这期间我们应该怎么办

2:如何做到负载均衡

3:如何有效的将数据发送到相关的接收者?就是怎么样过滤

4:如何保证消费者收到完整正确的数据

5:如何让优先级高的接收者先收到数据

一:"Hello RabbitMQ"

下面有一幅图,其中P表示生产者,C表示消费者,红色部分为消息队列

e803af0e3d4c6b2583f3f080ff8d8409.png

二:项目开始

2.1:首先引入rabbitMQ jar包

com.rabbitmq

amqp-client

3.6.5

2.2:创建消费者Producer

/*** 消息生成者*/

public classProducer {public final static String QUEUE_NAME="rabbitMQ.test";public static void main(String[] args) throwsIOException, TimeoutException {//创建连接工厂

ConnectionFactory factory = newConnectionFactory();//设置RabbitMQ相关信息

factory.setHost("localhost");//factory.setUsername("lp");//factory.setPassword("");//factory.setPort(2088);//创建一个新的连接

Connection connection =factory.newConnection();//创建一个通道

Channel channel =connection.createChannel();//声明一个队列 channel.queueDeclare(QUEUE_NAME, false, false, false, null);

String message= "Hello RabbitMQ";//发送消息到队列中

channel.basicPublish("", QUEUE_NAME, null, message.getBytes("UTF-8"));

System.out.println("Producer Send +'" + message + "'");//关闭通道和连接

channel.close();

connection.close();

}

}

注1:queueDeclare第一个参数表示队列名称、第二个参数为是否持久化(true表示是,队列将在服务器重启时生存)、第三个参数为是否是独占队列(创建者可以使用的私有队列,断开后自动删除)、第四个参数为当所有消费者客户端连接断开时是否自动删除队列、第五个参数为队列的其他参数

注2:basicPublish第一个参数为交换机名称、第二个参数为队列映射的路由key、第三个参数为消息的其他属性、第四个参数为发送信息的主体

2.3:创建消费者

public classCustomer {private final static String QUEUE_NAME = "rabbitMQ.test";public static void main(String[] args) throwsIOException, TimeoutException {//创建连接工厂

ConnectionFactory factory = newConnectionFactory();//设置RabbitMQ地址

factory.setHost("localhost");//创建一个新的连接

Connection connection =factory.newConnection();//创建一个通道

Channel channel =connection.createChannel();//声明要关注的队列

channel.queueDeclare(QUEUE_NAME, false, false, true, null);

System.out.println("Customer Waiting Received messages");//DefaultConsumer类实现了Consumer接口,通过传入一个频道,//告诉服务器我们需要那个频道的消息,如果频道中有消息,就会执行回调函数handleDelivery

Consumer consumer = newDefaultConsumer(channel) {

@Overridepublic voidhandleDelivery(String consumerTag, Envelope envelope,

AMQP.BasicProperties properties,byte[] body)throwsIOException {

String message= new String(body, "UTF-8");

System.out.println("Customer Received '" + message + "'");

}

};//自动回复队列应答 -- RabbitMQ中的消息确认机制

channel.basicConsume(QUEUE_NAME, true, consumer);

}

前面代码我们可以看出和生成者一样的,后面的是获取生产者发送的信息,其中envelope主要存放生产者相关信息(比如交换机、路由key等)body是消息实体。

2.4:运行结果

生产者:

1d2a6cd66366b17481edaf2bbeead4f0.png

消费者:

d238adfec1cc3b7d9c9d29d7d1eba67a.png

三:实现任务分发

工作队列

69651671e4fe7a9688b8fc677101fa42.png

一个队列的优点就是很容易处理并行化的工作能力,但是如果我们积累了大量的工作,我们就需要更多的工作者来处理,这里就要采用分布机制了。

我们新创建一个生产者NewTask

public classNewTask {private static final String TASK_QUEUE_NAME="task_queue";public static void main(String[] args) throwsIOException, TimeoutException {

ConnectionFactory factory=newConnectionFactory();

factory.setHost("localhost");

Connection connection=factory.newConnection();

Channel channel=connection.createChannel();

channel.queueDeclare(TASK_QUEUE_NAME,true,false,false,null);//分发信息

for (int i=0;i<10;i++){

String message="Hello RabbitMQ"+i;

channel.basicPublish("",TASK_QUEUE_NAME,

MessageProperties.PERSISTENT_TEXT_PLAIN,message.getBytes());

System.out.println("NewTask send '"+message+"'");

}

channel.close();

connection.close();

}

}

然后创建2个工作者Work1和Work2代码一样

public classWork1 {private static final String TASK_QUEUE_NAME = "task_queue";public static void main(String[] args) throwsIOException, TimeoutException {final ConnectionFactory factory = newConnectionFactory();

factory.setHost("localhost");

Connection connection=factory.newConnection();final Channel channel =connection.createChannel();

channel.queueDeclare(TASK_QUEUE_NAME,true, false, false, null);

System.out.println("Worker1 Waiting for messages");//每次从队列获取的数量

channel.basicQos(1);final Consumer consumer = newDefaultConsumer(channel) {

@Overridepublic voidhandleDelivery(String consumerTag,

Envelope envelope,

AMQP.BasicProperties properties,byte[] body) throwsIOException {

String message= new String(body, "UTF-8");

System.out.println("Worker1 Received '" + message + "'");try{throw newException();//doWork(message);

}catch(Exception e){

channel.abort();

}finally{

System.out.println("Worker1 Done");

channel.basicAck(envelope.getDeliveryTag(),false);

}

}

};boolean autoAck=false;//消息消费完成确认

channel.basicConsume(TASK_QUEUE_NAME, autoAck, consumer);

}private static voiddoWork(String task) {try{

Thread.sleep(1000); //暂停1秒钟

} catch(InterruptedException _ignored) {

Thread.currentThread().interrupt();

}

}

}

注:channel.basicQos(1);保证一次只分发一个 。autoAck是否自动回复,如果为true的话,每次生产者只要发送信息就会从内存中删除,那么如果消费者程序异常退出,那么就无法获取数据,我们当然是不希望出现这样的情况,所以才去手动回复,每当消费者收到并处理信息然后在通知生成者。最后从队列中删除这条信息。如果消费者异常退出,如果还有其他消费者,那么就会把队列中的消息发送给其他消费者,如果没有,等消费者启动时候再次发送。

dfa3fce4a1d6bc1bbb9714141fade637.png

72e701aa4f69d69dfa4810d9281c943c.png

9be7ee63859c7972d69b0ddb1d0cdf4c.png

关于上面我们遗留问题在下一篇继续讲解

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值