简介
远程过程调用(RPC): 客户端发送一个请求到远程服务器上,远程服务器接收请求并处理结果,将结果响应给客户端,这个过程被称为远程过程调用。
RPC涉及到的基本知识:
- 关于队列:整个过程会设计到两个队列一个是专门保存请求的队列,一般名字被称为rpc_queue,另一个队列被称为响应队列,专门用于保存服务器处理的响应结果,这个队列的名字是随机生成的字符串。
- 关于消息的基本属性BasicProperties:回复(replyTo):是响应队列的名字,当服务器接收请求并处理好结果,服务器需要知道将响应的信息发送到哪个队列中;关联id(correlationId):是一个UUID值,发消息的时候会带上这个值,该值在客户端接收响应时用于判断接收到的响应消息是否是自己发出请求对应的响应; 客户端在发送请求时需要带上replyTo和correlationId两个属性。
- 其他属性:contentType:内容类型,用来描述编码的MIME类型。例如,经常使用JSON编码是将此属性设置为一个很好的做法:application/json,在rpc中该属性不是必须的
RPC的过程描述:
- 客户端首先将请求消息发送到请求队列,在发送请求时需要指定replyTo和correlationId两个值;
- 服务端需要预先订阅请求队列(rpc_queue),以便服务器端能随时接受到请求消息,当服务端接收到请求消息时对请求进行处理,将处理结果发送到响应队列(随机队列)中
- 客户端还要预先订阅响应队列(随机队列),以便当服务器发送响应消息到响应队列中,客户端能及时收到响应结果,服务器在将响应发送到响应队列中还要指定correlationId值(类似于唯一标记当前的请求),这样当客户端从响应队列中接收到消息时就可以通过correlationId的值是否和发送请求的关联id值是否相同,如果相同就证明这个响应结果就是这个请求对应的响应结果。注意这个预先订阅响应队列的步骤需要在客户端中完成,最好在客户端发送请求消息前就完成。
server,client既是生产者又是消费者
Server:
package com.suirui.mq.rabbit.rpc;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
/**
* @Author zongx
* @Date 2020/5/9 14:23
* @Version 1.0
* 路由选择Routing
*/
public class Server {
public static void main(String[] args) throws IOException, TimeoutException {
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("127.0.0.1");
factory.setPort(AMQP.PROTOCOL.PORT);
factory.setUsername("guest");
factory.setPassword("guest");
Connection connection = factory.newConnection();
Channel channel = connection.createChannel();
String RPC_QUEUQU = "rpc_queue";
// String EXCHANGE_NAME = "exchange.rpc1";
String EXCHANGE_NAME ="";
channel.queueDeclare(RPC_QUEUQU, false, false, false, null);
// 声明交换机
// channel.exchangeDeclare(EXCHANGE_NAME, "direct");
channel.basicQos(1);
Consumer consumer = new DefaultConsumer(channel){
@Override
public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
//接收client发送信息
String message = new String(body, "UTF-8");
System.out.println("服务端:已接收到请求消息:" + message);
// 服务器端接收到消息并处理消息
String response = "{'code': 200, 'data': '" + message+ "'}";
// 将消息发布到reply_to响应队列中
// 获取返回queue名称
AMQP.BasicProperties replyProperties = new AMQP.BasicProperties.Builder().
correlationId(properties.getCorrelationId()).build();
String replyTo = properties.getReplyTo();
channel.basicPublish(EXCHANGE_NAME, replyTo, replyProperties, response.getBytes("UTF-8"));
System.out.println("服务端:请求已处理完毕,响应结果" + response + "已发送到响应队列中");
// 手动应答
channel.basicAck(envelope.getDeliveryTag(), false);
}
};
// 手动应答模式
channel.basicConsume(RPC_QUEUQU, false, consumer);
System.out.println("服务端:已订阅请求队列(rpc_queue), 开始等待接收请求消息...");
try {
Thread.sleep(100000);
} catch (InterruptedException e) {
e.printStackTrace();
}
channel.close();
connection.close();
}
}
Client
package com.suirui.mq.rabbit.rpc;
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.UUID;
import java.util.concurrent.TimeoutException;
/**
* @Author zongx
* @Date 2020/5/9 14:23
* @Version 1.0
* 路由选择Routing
*/
public class Client {
public static void main(String[] args) throws IOException, TimeoutException {
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("127.0.0.1");
factory.setPort(AMQP.PROTOCOL.PORT);
factory.setUsername("guest");
factory.setPassword("guest");
Connection connection = factory.newConnection();
Channel channel = connection.createChannel();
String EXCHANGE_NAME = "";
// 声明交换机
// channel.exchangeDeclare(EXCHANGE_NAME, "direct");
// 预先定义响应的结果,即预先订阅响应结果的队列,先订阅响应队列,再发送消息到请求队列
String reply_to_queue = channel.queueDeclare().getQueue();
final String correlationId = UUID.randomUUID().toString();
Consumer consumer = new DefaultConsumer(channel){
@Override
public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
//收取server的响应信息
String message = new String(body, "UTF-8");
System.out.println("已接收到服务器的响应结果:" + message);
}
};
//设置自动ack
channel.basicConsume(reply_to_queue, true, consumer);
//发送请求,将消息发送到请求队列中
String RPC_QUEUE = "rpc_queue";
String message = "Hello RabbitMQ";
AMQP.BasicProperties properties = new AMQP.BasicProperties.Builder().correlationId(correlationId).replyTo(reply_to_queue).build();
channel.basicPublish(EXCHANGE_NAME, RPC_QUEUE, properties, message.getBytes("UTF-8"));
System.out.println("已发出请求请求消息:" + message);
try {
Thread.sleep(100000);
} catch (InterruptedException e) {
e.printStackTrace();
}
channel.close();
connection.close();
}
}
运行效果
先运行服务端,再运行客户端