Remote Procedure Call:远程过程调用,一次远程过程调用的流程即客户端发送一个请求到服务端,服务端根据请求信息进行处理后返回响应信息,客户端收到响应信息后结束
为了说明如何使用RPC服务,我们将把“发送方”和“接收方”更改为“客户端”和“服务器”。当我们调用服务时,我们将得到我们对应的斐波那契值
Integer response = (Integer) template.convertSendAndReceive
(exchange.getName(), "rpc", 5);
System.out.println(" [.] Got '" + response + "'");
一般来说,通过RabbitMQ执行RPC很容易。客户端发送请求消息,服务器进行回复。为了接收响应,我们需要随请求一起发送“回调”队列地址。当我们使用上述convertSendAndReceive()方法时,Spring AMQP的RabbitTemplate为我们处理回调队列。使用RabbitTemplate时,无需执行任何其他设置
Message properties消息属性
Spring AMQP允许您关注正在使用的消息方式,并隐藏支持此方式所需的消息管道的详细信息。例如,本机客户端通常会为每个RPC请求创建一个回调队列。这效率很低,所以另一种方法是为每个客户端创建一个回调队列
这引发了一个新问题,在该队列中收到响应后,不清楚响应属于哪个请求。此时将使用correlationId属性。Spring AMQP自动为每个请求设置唯一值。此外,它还处理拥有正确correlationID的响应
Spring AMQP使RPC样式更容易的一个原因是,有时您可能希望忽略回调队列中的未知消息,而不是由于错误而失败。这是由于服务器端可能存在竞争条件。虽然不太可能,但RPC服务器可能会在向我们发送应答之后,但在发送请求的确认消息之前死亡。如果发生这种情况,重新启动的RPC服务器将再次处理该请求。Spring AMQP客户端优雅地处理重复的响应,理想情况下RPC应该是幂等的
RPC工作流程如下:
@Configuration
public class RpcConfig {
@Bean
public DirectExchange rpc() {
return new DirectExchange("rpc");
}
private static class ServerConfig {
@Bean
public Queue rpcQueue() {
return new Queue("rpc.requests");
}
@Bean
public Binding binding(DirectExchange rpc,
Queue rpcQueue) {
return BindingBuilder.bind(rpcQueue)
.to(rpc)
.with("rpc");
}
}
}
@Component
public class RpcClient {
private RabbitTemplate rabbitTemplate;
public RpcClient(RabbitTemplate rabbitTemplate) {
this.rabbitTemplate = rabbitTemplate;
}
public void send() {
System.out.println(" [客户端] 发送请求10");
Integer response = (Integer) rabbitTemplate.convertSendAndReceive("rpc", "rpc", 10);
System.out.println(" [客户端] 收到响应 " + response + "");
}
}
@Component
public class RpcServer {
@RabbitListener(queues = "rpc.requests")
public int fibonacci(int n) {
System.out.println(" [服务端] 收到请求" + n);
int result = fib(n);
System.out.println(" [服务端] 响应请求 " + result);
return result;
}
public int fib(int n) {
return n == 0 ? 0 : n == 1 ? 1 : (fib(n - 1) + fib(n - 2));
}
}
@SpringBootTest
public class RabbitTest {
@Autowired
private RpcClient rpcClient;
@Test
public void testRpc() {
rpcClient.send();
}
欢迎关注公众号算法小生或沈健的技术博客shenjian.online