news 2026/6/7 7:13:00

GetDirectoryReference 远程代理创建机制详解

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
GetDirectoryReference 远程代理创建机制详解

GetDirectoryReference 远程代理创建机制详解

一、GetDirectoryReference 方法概述

internalIRemoteGrainDirectoryGetDirectoryReference(SiloAddresssilo){returnthis.grainFactory.GetSystemTarget<IRemoteGrainDirectory>(Constants.DirectoryServiceType,silo);}

这个方法的核心作用是:

  • 创建一个指向目标 Silo 的IRemoteGrainDirectory接口远程代理
  • 该代理会将方法调用自动转换为网络消息发送到目标 Silo

二、远程代理创建的完整流程

1. 第一步:调用 GrainFactory.GetSystemTarget 方法

publicTGrainInterfaceGetSystemTarget<TGrainInterface>(GrainTypegrainType,SiloAddressdestination)whereTGrainInterface:ISystemTarget{vargrainId=SystemTargetGrainId.Create(grainType,destination);returnthis.GetSystemTarget<TGrainInterface>(grainId.GrainId);}
关键操作:
  • 使用SystemTargetGrainId.Create创建一个特殊的GrainId
  • 这个GrainId包含了目标服务类型目标 Silo 地址

2. 第二步:创建 SystemTargetGrainId

// 在 GrainDirectoryPartition.cs 中的调用示例internalstaticSystemTargetGrainIdCreateGrainId(SiloAddresssiloAddress,intpartitionIndex)=>SystemTargetGrainId.Create(Constants.GrainDirectoryPartitionType,siloAddress,partitionIndex.ToString(CultureInfo.InvariantCulture));
设计意图:
  • SystemTargetGrainId是一种特殊的 GrainId
  • 它将目标 Silo 地址编码到 GrainId 中,确保请求被路由到正确的 Silo
  • 不需要依赖 Grain 目录来定位目标,因为 Silo 地址已经包含在 GrainId 中

3. 第三步:调用重载的 GetSystemTarget 方法

publicTGrainInterfaceGetSystemTarget<TGrainInterface>(GrainIdgrainId)whereTGrainInterface:ISystemTarget{ISystemTargetreference;ValueTuple<GrainId,Type>key=ValueTuple.Create(grainId,typeof(TGrainInterface));lock(this.typedSystemTargetReferenceCache){if(this.typedSystemTargetReferenceCache.TryGetValue(key,outreference)){return(TGrainInterface)reference;}reference=this.GetGrain<TGrainInterface>(grainId);this.typedSystemTargetReferenceCache[key]=reference;return(TGrainInterface)reference;}}
关键操作:
  • 缓存检查:首先检查是否已经存在相同的代理引用
  • 创建新代理:如果缓存中没有,调用this.GetGrain<TGrainInterface>(grainId)创建新代理
  • 缓存新代理:将新创建的代理缓存起来以便重用

4. 第四步:调用 GetGrain 方法创建代理

publicTGrainInterfaceGetGrain<TGrainInterface>(GrainIdgrainId)whereTGrainInterface:IAddressable{return(TGrainInterface)this.CreateGrainReference(typeof(TGrainInterface),grainId);}publicIAddressableGetGrain(GrainIdgrainId,GrainInterfaceTypeinterfaceType){returnthis.referenceActivator.CreateReference(grainId,interfaceType);}
核心组件:
  • referenceActivatorGrainReferenceActivator的实例
  • 它负责创建Grain引用代理对象

5. 第五步:GrainReferenceActivator.CreateReference 方法

// 在 SystemTarget.cs 中的使用示例publicGrainReferenceGrainReference=>_selfReference??=_shared.GrainReferenceActivator.CreateReference(_id.GrainId,default);
工作原理:
  • 根据GrainIdGrainInterfaceType创建一个代理对象
  • 这个代理对象实现了TGrainInterface接口(在这里是IRemoteGrainDirectory
  • 代理对象内部包含了消息发送逻辑,会将方法调用转换为网络消息

三、远程代理的工作机制

1. 代理的本质

创建的远程代理是一个动态生成的类,它:

  • 实现了IRemoteGrainDirectory接口
  • 重写了接口中的所有方法(如RegisterAsyncUnregisterAsync等)
  • 在每个方法中包含了消息序列化网络发送逻辑

2. 方法调用的转换

当调用代理的方法时(如remoteProxy.RegisterAsync(grainAddress)):

  1. 代理将方法名参数值序列化为二进制数据
  2. 创建一个包含这些数据的消息对象
  3. 设置消息的目标地址为包含在GrainId中的 Silo 地址
  4. 通过 Orleans 的消息传递系统将消息发送到目标 Silo
  5. 等待目标 Silo 的响应消息
  6. 将响应消息反序列化为返回值,返回给调用者

3. 消息传递系统

Orleans 的消息传递系统负责:

  • 消息的路由传递
  • 网络连接的维护复用
  • 消息的序列化反序列化
  • 超时和重试机制
  • 负载均衡和故障转移

四、关键设计点

1. 缓存机制

lock(this.typedSystemTargetReferenceCache){if(this.typedSystemTargetReferenceCache.TryGetValue(key,outreference)){return(TGrainInterface)reference;}// ... 创建新代理并缓存}
设计意图:
  • 避免重复创建相同的代理对象
  • 提高性能,减少内存开销
  • 确保对同一目标的多次调用使用相同的代理实例

2. SystemTarget 的特殊性

SystemTarget 与普通 Grain 的区别:

  • 更轻量级:不需要激活和生命周期管理
  • 确定性地址:地址由 Silo 地址和服务类型确定
  • 不依赖 Grain 目录:不需要注册到 Grain 目录
  • 更高性能:避免了 Grain 激活的开销

3. 类型安全

publicTGrainInterfaceGetSystemTarget<TGrainInterface>(GrainTypegrainType,SiloAddressdestination)whereTGrainInterface:ISystemTarget
设计意图:
  • 确保返回的代理实现了指定的接口
  • 提供编译时类型检查
  • 避免运行时类型转换错误

五、实际调用示例

假设我们有两个 Silo:Silo1 和 Silo2

// 在 Silo1 中执行 SiloAddress silo2Address = SiloAddress.FromParsableString("192.168.1.100:11111"); IRemoteGrainDirectory remoteProxy = GetDirectoryReference(silo2Address); // 这个调用会被自动转换为网络消息发送到 Silo2 await remoteProxy.RegisterAsync(grainAddress, null, 1);

在 Silo2 上的处理:

  1. 接收网络消息
  2. 反序列化消息内容
  3. 找到本地的RemoteGrainDirectory实例
  4. 调用本地RegisterAsync方法
  5. 将结果序列化并发送回 Silo1

六、代码关系图

GetDirectoryReference(silo) └─→ GrainFactory.GetSystemTarget<IRemoteGrainDirectory>(grainType, silo) ├─→ SystemTargetGrainId.Create(grainType, silo) // 创建包含目标地址的 GrainId └─→ GetSystemTarget<IRemoteGrainDirectory>(grainId) ├─→ 检查缓存 (typedSystemTargetReferenceCache) │ └─→ 缓存命中 → 返回现有代理 └─→ 缓存未命中 → 创建新代理 ├─→ GetGrain<IRemoteGrainDirectory>(grainId) │ └─→ CreateGrainReference(typeof(IRemoteGrainDirectory), grainId) └─→ referenceActivator.CreateReference(grainId, interfaceType) └─→ 创建包含消息发送逻辑的代理对象 └─→ 缓存新代理 └─→ 返回代理对象

七、总结

GetDirectoryReference 方法创建远程代理的机制是 Orleans 分布式通信的核心:

  1. 透明的分布式编程模型:调用远程方法就像调用本地方法一样
  2. 高效的消息传递:自动处理网络通信、序列化和反序列化
  3. 类型安全:提供编译时类型检查,避免运行时错误
  4. 性能优化:通过缓存机制避免重复创建代理
  5. 可靠性:内置重试、超时和故障转移机制

这种设计使得开发人员可以专注于业务逻辑,而无需关心底层的网络通信细节,体现了 Orleans 的核心设计理念:让分布式编程变得简单

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/5 2:21:40

使用 C# 高质量实现 Word 转 Excel:完整保留文本、表格与样式

在企业办公和文档自动化场景中&#xff0c;Word 和 Excel 是最常用的两种文档格式。Word 更适合撰写说明性文本和排版复杂的内容&#xff0c;而 Excel 则擅长结构化数据处理、统计与分析。在实际开发中&#xff0c;我们经常会遇到这样的需求&#xff1a; 将 Word 文档中的内容&…

作者头像 李华
网站建设 2026/6/3 22:42:28

c++进程池(Linux)的实现(2025.12.22)

学习比特课程后&#xff0c;学习总结和代码实现。这节课产生了两点困惑&#xff0c;查阅资料后&#xff0c;有一下理解&#xff1a;1.“FD 数值相同”≠“指向同一个管道”比如两次pipe()可能都生成pipefd[0]3&#xff08;因为前一轮父进程关闭了读端 3&#xff0c;FD 号被复用…

作者头像 李华
网站建设 2026/6/5 21:06:54

多通道小动物代谢监控系统 小动物代谢监测系统 小动物代谢检测系统

小动物代谢系统(AMMS)具有实时统计、自动化等优点&#xff0c;提高了药物研发和基础生命科学研究的效率&#xff0c;减少手工操作带来数据偏差及误差。, 在动物无拘束状态下&#xff0c;进行多通道测量Amms能实时统计大小鼠的饮食量、饮水量、运动量3个指标饮食槽设计防止粪…

作者头像 李华
网站建设 2026/6/7 5:57:20

AI论文助手推荐:8个高效工具实现自动降重与内容优化

在AI论文辅助工具领域&#xff0c;多款平台各具特色&#xff0c;涵盖降重、AIGC检测优化及论文撰写等核心功能。通过对8款主流工具的横向评测&#xff0c;结合实际测试结果与用户体验反馈&#xff0c;以下为综合性能排名&#xff08;评估维度包括处理效率、内容准确性及操作便捷…

作者头像 李华
网站建设 2026/6/4 18:26:23

动态规划(六)——分治优化DP 算法设计与分析 国科大

本文内容紧接动态规划&#xff08;五&#xff09;&#xff0c;讨论如何优化序列对齐算法Hirschberg算法上文最后提到的解决方案&#xff0c;是维护一个OPT矩阵&#xff0c;那么它的空间开销就变成了O(mn)&#xff0c;而Hirschberg 算法通过分治策略&#xff0c;将序列对齐问题的…

作者头像 李华
网站建设 2026/6/3 14:30:31

基于python的网上商城比价系统(源码+vue+前后端分离)

前言基于Python的网上商城比价系统是一种先进的应用程序&#xff0c;旨在帮助消费者在众多在线商城中快速找到性价比最高的商品。以下是对该系统的详细介绍&#xff1a; 一、系统背景与意义 随着电子商务的迅猛发展&#xff0c;网上商城已成为消费者购物的主要渠道。然而&#…

作者头像 李华