RabbitMQ九:远程过程调用RPC
定义
RPC(Remote Procedure Call Protocol)——远程过程调用协议:它是一种通过网络从远程计算机程序上请求服务,而不需要了解底层网络技术的协议。RPC协议假定某些传输协议的存在,如TCP或UDP,为通信程序之间携带信息数据。在OSI网络通信模型中,RPC跨越了传输层和应用层。RPC使得开发包括网络分布式多程序在内的应用程序更加容易。
PRC采用客户端/服务端模式,请求程序就是一个客户机,而服务提供就是一个服务器。首先,客户机调用进程发送一个有进程参数的调用信息到服务进程,然后等待应答信息,在服务器,进程保持睡眠状态直到调用信息到达为止。当一个调用信息到达,服务器获得进程参数,计算结果,发送答复消息,然后等待下一个调用信息,最后,客户端调用进程接收答复信息,获得进程结果,然后调用执行继续前进。(整个过程有点类似:你到某大医院看病,你先到柜台交钱拿卡(医师费),拿卡去找医生(卡代表你的认证相当参数),医生根据卡给你把脉看病进行详谈沟通,医诊结束后给你开药,下一位患者进入。。。。。,医生有是服务端,患者是客户端,举例可能有点牵强,就是表达那个意思,)
RPC是在计算机中一种常见的模式,是通常我要用消息队列3个关键点:
、服务的寻址; 、消息的接受; 、消息的关联。
RPC调用的顺序简述:
、当客户端启动时,它会创建一个匿名的独占会回调队列; 、对于一个RPC请求,客户端通过两个属性发送一条消息(从图中我们也可以看到):relayTo 设置回调队列;correlationId,为每个请求设置唯一的标识ID; 、消息将发送到一个Rpc_queue 队列; 、RPC工程线程(服务器)在该队列上等待请求,当请求出现,他将处理请求并把结果发回到客户端,使用队列在replayTo中设置; 、客户端在回调队列上等待响应,当消息出现,它检查关联ID,如果匹配来自请求的关联ID值,返回队列消息到该应用程序。
重点解释
correlationId 和 relayTo 参数
首先客户端通过RPC向服务端发送请求
我这里有一堆东西需要你给我处理一下,correlationId :这是我的请求标识,relayTo :你处理完过后把结果返回到这个队列中。
服务端拿到请求,并开始处理并返回结果
correlationId :这是你的请求标识 ,原封不动的给你。 这时候客户端用自己的correlationId 与服务端返回的id进行对比。是我的,就接收。
适合RPC场合说明
希望同步得到数据的场合,RPC合适; 希望使用简单,则RPC;RPC操作基于接口,使用简单,使用方式模式本地调用。异步的方式编程比较复杂。 不希望客户端受限于服务端的速度等,可以使用Message Queue
RabbitMQ RPC的特点
Message Queue 把所有的请求消息存储起来,然后处理,和客户端解耦; Message Queue 引用新的结点,系统的可靠性会受Message Queue 结点的影响; Meaage Queue 是异步单向的消息,发送消息设计成是不需要等待消息处理的完成。 所以对于有同步返回需求,Message Queue 是个不错的方向
普通RPC的特点
同步调用,对于要等待返回结果、处理结果的场景,RPC是可以非常自然直觉的使用方式,当然RPC也可以异步调用。 由于等待结果,客户端会有线程消耗。 如果以异步RPC的方式使用,客户端线程消耗可以去掉,但不能做到像消息一样暂存消息请求,压力会直接传导到服务端。
代码块
备注(创建两个解决方案:服务端和客户端)
服务端
static void Main(string[] args)
{
using (var channel = GetConnection().CreateModel())
{
channel.QueueDeclare("rpc_queue", true, false, false, null);
channel.BasicQos(, , false);
var consumer = new EventingBasicConsumer(channel);
// var consumer = new QueueingBasicConsumer(channel);
channel.BasicConsume("rpc_queue", false, consumer);
Console.WriteLine("等待 RPC 队列");
consumer.Received += (model, ea) =>
{
// while (true)
// {
string response = null;
//出列
// var ea = (BasicDeliverEventArgs)consumer.Queue.Dequeue();
var body = ea.Body;
var props = ea.BasicProperties;
//内容的基本属性
var replyProps = channel.CreateBasicProperties();
//注意这里的correlationId
replyProps.CorrelationId = props.CorrelationId;
try
{
var message = Encoding.UTF8.GetString(body);
int n = int.Parse(message);
Console.WriteLine("显示内容" + message);
response = fib(n).ToString();
}
catch (Exception e)
{ Console.WriteLine("报错" + e.ToString());
response = "";
}
finally
{
var responseBytes = Encoding.UTF8.GetBytes(response);
channel.BasicPublish("", props.ReplyTo, replyProps, responseBytes);
channel.BasicAck(ea.DeliveryTag, false);
}
// };
};
Console.WriteLine("发布成功!!!");
Console.ReadLine();
}
} /// <summary>
/// 私有方法
/// </summary>
/// <param name="n"></param>
/// <returns></returns>
private static int fib(int n)
{
if (n == || n == )
{
return n;
}
//Thread.Sleep()方法用于将当前线程休眠一定时间 时间单位是毫秒 1000毫秒= 1秒
//System.Threading.Thread.Sleep(2000);当前休眠2秒
//suspen()挂起当前线程。也可以指定挂起时间。
//close() 关闭当前线程。
Thread.Sleep( * );
return n;
// return fib(n - 1) + fib(n - 2);
}
客户端(两个类:Consumer,HelpConnection)
Consumer代码块:
static void Main(string[] args)
{
for (int i = ; i < ; i++)
{
Stopwatch watch = new Stopwatch();
watch.Start();
var rpcClient = new HelpConnection();
Console.WriteLine("显示内容" + i.ToString());
var response = rpcClient.Call(i);
Console.WriteLine("显示内容" + response);
//当前连接关闭
rpcClient.Close();
watch.Stop();
Console.WriteLine(string.Format(" [x] Requesting complete {0} ,cost {1} ms", i, watch.Elapsed.TotalMilliseconds));
}
Console.WriteLine(" complete!!!! ");
Console.ReadLine();
}
HelpConnection代码块:
/// <summary>
/// 成员变量
/// </summary>
private static IConnection connection { get; set; }
private IModel channel { get; set; }
private string replyQueueName { get; set; }
private QueueingBasicConsumer consumer { get; set; } /// <summary>
/// 构造方法:连接配置
/// </summary>
public HelpConnection()
{
var factory = new ConnectionFactory()
{
//计算机名称,账号,密码,
HostName = "localhost",
UserName = "zhangguangpo",
Password = "guangpo1992",
//RequestedHeartbeat = 60,
AutomaticRecoveryEnabled = true //要启用自动连接恢复
};
//创建连接
connection = factory.CreateConnection();
channel = connection.CreateModel();
//而客户端为了获得处理结果,那么客户在向服务器发送请求时,同时发送一个回调队列地址reply_to
replyQueueName = channel.QueueDeclare().QueueName;
consumer = new QueueingBasicConsumer(channel);
channel.BasicConsume(queue: replyQueueName,
noAck: true,
consumer: consumer);
// return Connection;
} /// <summary>
/// 消息判断
/// </summary>
/// <param name="message"></param>
/// <returns></returns>
public string Call(int message)
{
var corrId = Guid.NewGuid().ToString();
var props = channel.CreateBasicProperties();
props.ReplyTo = replyQueueName;
props.CorrelationId = corrId;
var messageBates = Encoding.UTF8.GetBytes(message.ToString());
channel.BasicPublish("", "rpc_queue", props, messageBates);
while (true)
{
var ea = (BasicDeliverEventArgs)consumer.Queue.Dequeue();
if (ea.BasicProperties.CorrelationId == corrId)
{
var body = Encoding.UTF8.GetString(ea.Body);
return body;
}
}
} /// <summary>
/// 当前连接关闭
/// </summary>
public void Close()
{
connection.Close();
}
效果图
- 博主是利用读书、参考、引用、抄袭、复制和粘贴等多种方式打造成自己的纯镀 24k 文章,请原谅博主成为一个无耻的文档搬运工!
- 小弟刚迈入博客编写,文中如有不对,欢迎用板砖扶正,希望给你有所帮助。
RabbitMQ九:远程过程调用RPC的更多相关文章
- RabbitMQ 实现远程过程调用RPC
RPC调用的顺序1. 在客户端初始化的时候,也就是SimpleRpcClient类初始化的时候,它会随机的创建一个callback队列,用于存放服务的返回值,这个队列是exclusive的.连接断开就 ...
- RabbitMQ入门教程(八):远程过程调用RPC
原文:RabbitMQ入门教程(八):远程过程调用RPC 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https://blog.cs ...
- python中RabbitMQ的使用(远程过程调用RPC)
在RabbitMQ消息队列中,往往接收者.发送者不止是一个身份.例如接接收者收到消息并且需要返回给发送者. 此时接收者.发送者的身份不再固定! 我们来模拟该情形: 假设有客户端client,服务端se ...
- Thrift架构~从图中理解thrift,它事实上是一种远程过程调用
thrift为我们简化了tcp通讯,它可以使用我们方便的建立各种语言的服务端与客户端,并实现客户端对服务器的远程过程调用,简单的说就是服务器通过thrift架构对外开放一些接口,并自己实现这些接口,如 ...
- PRC远程过程调用
RPC(Remote Promote Call) 一种进程间通信方式.允许像调用本地服务一样调用远程服务. RPC框架的主要目标就是让远程服务调用更简单.透明.RPC框架负责屏蔽底层的传输方式(TCP ...
- rabbitMQ学习笔记(七) RPC 远程过程调用
关于RPC的介绍请参考百度百科里的关于RPC的介绍:http://baike.baidu.com/view/32726.htm#sub32726 现在来看看Rabbitmq中RPC吧!RPC的工作示意 ...
- RabbitMQ学习系列(五): RPC 远程过程调用
前面讲过一些RabbitMQ的安装和用法,也说了说RabbitMQ在一般的业务场景下如何使用.不知道的可以看我前面的博客,http://www.cnblogs.com/zhangweizhong/ca ...
- RPC远程过程调用
什么是RPC: 将一个函数运行在远程计算机上并且等待获取那里的结果,这个称作RPC: (Remote Procedure Call远程过程调用) RPC是一个计算机通信协议. rpc指的是在计算机A上 ...
- rabbitMQ学习3-RPC远程过程调用
将一个函数运行在远程计算机上并且等待获取那里的结果,这个称作远程过程调用(Remote Procedure Call)或者 RPC. RPC是一个计算机通信协议. 比喻 将计算机服务运行理解为厨师做饭 ...
随机推荐
- 怎样在QML中利用Sprite来做我们须要的动画
在游戏中动画的设计很中要. 在QML中,它提供了丰富的animation.可是有时我们须要对图像进行变化,就像放电影一样.在今天的这篇文章中,我们将设计一个能够变化图像的动画. 我们能够通过Qt所提供 ...
- POJ3984 迷宫问题 —— BFS
题目链接:http://poj.org/problem?id=3984 迷宫问题 Time Limit: 1000MS Memory Limit: 65536K Total Submissions ...
- 转:Oracle客户端NLS_LANG参数的设置详解
原文:http://database.51cto.com/art/201107/279361.htm 我们知道,Oracle客户端语言支持可以通过NLS_LANG参数的设置来完成,不同的系统平台上NL ...
- apt-get update 问题 及gcc高亮
一 gcc高亮 gcc 高亮有好几种方法. 参考 http://www.cokco.cn/thread-39909-1-1.html 这个教程: (1) git clone https://githu ...
- 棋盘问题(dp)
棋盘问题 传送门 题目描述 在一个给定形状的棋盘(形状可能是不规则的)上面摆放棋子,棋子没有区别.要求摆放时任意的两个棋子不能放在棋盘中的同一行或者同一列,请编程求解对于给定形状和大小的棋盘,摆放k个 ...
- Excel: 应用Match/Vlookup比较Excel两列的不同数据
假设Excel中有两列,现在要比较两列数据的不同.
- 关于spring boot在启动的时候报错: java.lang.Error: generate operation swagger failed, xxx.xxx.xxx
Error starting ApplicationContext. To display the auto-configuration report re-run your application ...
- $(function() {}),即$(document).ready(function(),什么时候执行?以此为准
$(function() { //执行操作 }); $(function() {}) 是$(document).ready(function()的简写. 这个函数什么时候执行的呢? 答案:DOM加载完 ...
- liteos 中断机制(四)
1. 概述 中断是指出现需要时, CPU暂停执行当前程序,转而执行新程序的过程.即在程序运行过程中,系统出现了一个必须由CPU立即处理的事务,此时, CPU暂时中止当前程序的执行转而处理这个事务,这个 ...
- (水题)洛谷 - P1464 - Function
https://www.luogu.org/problemnew/show/P1464 #include<bits/stdc++.h> using namespace std; #defi ...