Orleans Stream SubscriptionId 生成机制详解

概述

在 Orleans 流系统中,subscriptionId 是用于标识流订阅的唯一标识符。它有两种类型:显式订阅隐式订阅,每种类型都有不同的生成机制。

1. 订阅类型分类

显式订阅 (Explicit Subscription)

  • 通过代码显式调用 SubscribeAsync() 方法创建的订阅
  • 使用随机生成的 GUID
  • 在 GUID 的最后字节的高位设置为 0

隐式订阅 (Implicit Subscription)

  • 基于 Grain 类型和流命名空间自动匹配的订阅
  • 使用确定性算法生成 GUID
  • 在 GUID 的最后字节的高位设置为 1

2. 生成流程

2.1 显式订阅 ID 生成

csharp 复制代码
// 在 GrainBasedPubSubRuntime.cs 中
public GuidId CreateSubscriptionId(QualifiedStreamId streamId, GrainId streamConsumer)
{
    // 1. 生成随机 GUID
    Guid subscriptionId = Guid.NewGuid();
    
    // 2. 标记为显式订阅(清除最后字节的高位)
    subscriptionId = SubscriptionMarker.MarkAsExplicitSubscriptionId(subscriptionId);
    
    // 3. 包装为 GuidId
    return GuidId.GetGuidId(subscriptionId);
}

2.2 隐式订阅 ID 生成

csharp 复制代码
// 在 ImplicitStreamSubscriberTable.cs 中
private Guid MakeSubscriptionGuid(GrainType grainType, QualifiedStreamId streamId)
{
    Span<byte> bytes = stackalloc byte[16];
    
    // 1. 使用 Grain 类型的哈希码
    BinaryPrimitives.WriteUInt32LittleEndian(bytes, grainType.GetUniformHashCode());
    
    // 2. 使用 StreamId 的哈希码
    BinaryPrimitives.WriteUInt32LittleEndian(bytes[4..], streamId.StreamId.GetUniformHashCode());
    
    // 3. 使用 StreamId 的键索引
    BinaryPrimitives.WriteUInt32LittleEndian(bytes[8..], streamId.StreamId.GetKeyIndex());
    
    // 4. 使用 Provider 名称的稳定哈希
    BinaryPrimitives.WriteUInt32LittleEndian(bytes[12..], StableHash.ComputeHash(streamId.ProviderName));
    
    // 5. 标记为隐式订阅(设置最后字节的高位)
    return SubscriptionMarker.MarkAsImplictSubscriptionId(new(bytes));
}

3. 订阅标记机制

3.1 SubscriptionMarker 类

csharp 复制代码
internal static class SubscriptionMarker
{
    // 标记为显式订阅:清除最后字节的高位 (0x7F)
    internal static Guid MarkAsExplicitSubscriptionId(Guid subscriptionGuid)
    {
        return MarkSubscriptionGuid(subscriptionGuid, false);
    }
    
    // 标记为隐式订阅:设置最后字节的高位 (0x80)
    internal static Guid MarkAsImplictSubscriptionId(Guid subscriptionGuid)
    {
        return MarkSubscriptionGuid(subscriptionGuid, true);
    }
    
    // 检查是否为隐式订阅
    internal static bool IsImplicitSubscription(Guid subscriptionGuid)
    {
        Span<byte> guidBytes = stackalloc byte[16];
        subscriptionGuid.TryWriteBytes(guidBytes);
        // 检查最后字节的高位是否设置
        return (guidBytes[15] & 0x80) != 0;
    }
}

4. 完整生成流程

4.1 显式订阅流程

  1. 用户调用订阅方法

    csharp 复制代码
    await stream.SubscribeAsync(observer);
  2. StreamConsumer 创建订阅 ID

    csharp 复制代码
    // 在 StreamConsumer.cs 中
    GuidId subscriptionId = pubSub.CreateSubscriptionId(stream.InternalStreamId, myGrainReference.GetGrainId());
  3. GrainBasedPubSubRuntime 生成 ID

    csharp 复制代码
    Guid subscriptionId = SubscriptionMarker.MarkAsExplicitSubscriptionId(Guid.NewGuid());
  4. 设置观察者

    csharp 复制代码
    var subscriptionHandle = myExtension.SetObserver(subscriptionId, stream, observer, batchObserver, token, filterData);

4.2 隐式订阅流程

  1. 系统检查隐式订阅资格

    csharp 复制代码
    // 在 ImplicitStreamPubSub.cs 中
    if (!implicitTable.TryGetImplicitSubscriptionGuid(grainId, streamId, out subscriptionGuid))
    {
        throw new ArgumentOutOfRangeException(streamId.ToString(), "Only implicit subscriptions are supported.");
    }
  2. ImplicitStreamSubscriberTable 生成确定性 ID

    csharp 复制代码
    // 基于 Grain 类型、StreamId 和 Provider 名称生成确定性 GUID
    subscriptionId = MakeSubscriptionGuid(grainId.Type, streamId);
  3. 标记为隐式订阅

    csharp 复制代码
    return SubscriptionMarker.MarkAsImplictSubscriptionId(new(bytes));

5. 关键特性

5.1 唯一性保证

  • 显式订阅 :使用 Guid.NewGuid() 保证全局唯一性
  • 隐式订阅:使用确定性算法,相同输入产生相同输出

5.2 类型识别

  • 通过 GUID 最后字节的高位区分订阅类型
  • 0x80 = 隐式订阅,0x7F = 显式订阅

5.3 性能优化

  • 隐式订阅使用确定性算法,避免重复计算
  • 使用 Span<byte>stackalloc 减少内存分配

6. 使用场景

6.1 显式订阅

  • 用户主动订阅特定流
  • 需要精确控制订阅行为
  • 支持取消订阅

6.2 隐式订阅

  • 基于命名空间自动匹配
  • 简化开发者的订阅逻辑
  • 适用于广播场景

7. 总结

Orleans 的 subscriptionId 生成机制设计精巧,通过不同的生成策略和标记机制,既保证了唯一性,又支持了显式和隐式两种订阅模式。这种设计在保证性能的同时,提供了灵活的流订阅管理能力。

相关推荐
豆沙沙包?7 分钟前
2026年--Lc334-2130. 链表最大孪生和(链表转数组)--java版
java·数据结构·链表
柒.梧.15 分钟前
SSM常见核心面试问题深度解析
java·spring·面试·职场和发展·mybatis
杨章隐31 分钟前
Java 解析 CDR 文件并计算图形面积的完整方案(支持 MultipartFile / 网络文件)@杨宁山
java·开发语言
Renhao-Wan43 分钟前
Java 并发基石:AQS (AbstractQueuedSynchronizer)
java·开发语言
zlp19921 小时前
xxl-job java.sql.SQLException: interrupt问题排查(二)
java·开发语言
sunnyday04261 小时前
深入理解Java日志框架:Logback与Log4j2配置对比分析
java·log4j·logback
浩瀚地学1 小时前
【Java】异常
java·开发语言·经验分享·笔记·学习
张np1 小时前
java基础-LinkedHashMap
java·开发语言
毕设源码-朱学姐1 小时前
【开题答辩全过程】以 高校图书馆管理系统为例,包含答辩的问题和答案
java
xie_pin_an2 小时前
C++ 从入门到进阶:核心知识与实战指南
java·c++·算法