【问题标题】:Protobuf-net - How to use oneofProtobuf-net - 如何使用 oneof
【发布时间】:2021-10-20 14:14:16
【问题描述】:

我快速搜索了Protobuf-netoneof 的用法,它似乎是supported as of v2.3.0,但我一生都找不到任何关于如何的例子有人会使用它!

我的要求很简单,也许这也可以用[ProtoInclude] 解决,但我不太确定这到底是如何工作的。我有以下课程:

[ProtoContract]
public class ProgressUIMessage
{
    [ProtoMember(1)]
    public int Id {get; set;}

    [ProtoMember(2)]
    public object Message {get; set;}
}

Message 可以是 8 种不同的已知类型中的一种。这些类型根本不相互继承,虽然代码可以更改,但所有类型都没有共同点。

使用Google.Protobuf 我希望做一些事情similar to this,其中我有一个名为Instrument 的属性,它可以是上面示例中的两种类型之一,然后使用InstrumentOneofCase 来确定我的类型被给予。但是我如何在Protobuf-net 中实现同样的目标?

编辑: 我将保留最初的问题,但也许更多人可以涉及的更好的问题是:您将如何实现与 Protobuf-net 中的this MS example 相同的事情?无论是在编写类本身方面还是在确定参数最终是什么具体类型方面?

【问题讨论】:

    标签: .net grpc protobuf-net protobuf-net.grpc


    【解决方案1】:

    解决这个问题的方法是从您引用的 MS 示例中获取消息,并通过 protogen 运行它以查看它的作用 - 我们可以在这里非常方便地做到这一点:https://protogen.marcgravell.com/(注意我正在添加 @ 987654322@在文件顶部,MS示例中省略)。

    除其他外,这给了我们:

        [global::ProtoBuf.ProtoMember(2, Name = @"stock")]
        public Stock Stock
        {
            get => __pbn__instrument.Is(2) ? ((Stock)__pbn__instrument.Object) : default;
            set => __pbn__instrument = new global::ProtoBuf.DiscriminatedUnionObject(2, value);
        }
        public bool ShouldSerializeStock() => __pbn__instrument.Is(2);
        public void ResetStock() => global::ProtoBuf.DiscriminatedUnionObject.Reset(ref __pbn__instrument, 2);
    
        private global::ProtoBuf.DiscriminatedUnionObject __pbn__instrument;
    
        [global::ProtoBuf.ProtoMember(3, Name = @"currency")]
        public Currency Currency
        {
            get => __pbn__instrument.Is(3) ? ((Currency)__pbn__instrument.Object) : default;
            set => __pbn__instrument = new global::ProtoBuf.DiscriminatedUnionObject(3, value);
        }
        public bool ShouldSerializeCurrency() => __pbn__instrument.Is(3);
        public void ResetCurrency() => global::ProtoBuf.DiscriminatedUnionObject.Reset(ref __pbn__instrument, 3);
    

    所以我们可以看到它基本上使用了建立在DiscriminatedUnionObject 类型之上的条件序列化。实际上有一堆名为 DiscriminatedUnion* 的相关类型 - 取决于您需要重叠的内容,但由于它们都是这里的消息类型:DiscriminatedUnionObject 适合我们。


    还有一个可选的“oneof 应该使用枚举”选项(奇怪的是,“选项”下),如果启用,还会添加:

        public InstrumentOneofCase InstrumentCase => (InstrumentOneofCase)__pbn__instrument.Discriminator;
    
        public enum InstrumentOneofCase
        {
            None = 0,
            Stock = 2,
            Currency = 3,
        }
    

    否则,您将不得不使用ShouldSerialize*() 方法来解决活动案例。

    希望这能阐明 oneof 如何与 protobuf-net 一起使用。

    【讨论】:

    • 感谢 Marc,但您能否解释一下如何使用 protobuf-net 中的属性编写该类?我的意思是我想有比这更简单的方法:)
    • @cogumel0 最终,这里的目标之一是允许重叠存储,这样我们就不会拥有臃肿的数据类型,而做到这一点的唯一方法是通过DiscriminatedUnion* 的工作方式;您可以手动对其进行编码,但如上所示的代码广泛是使用 protobuf-net 实现“oneof”的预期方式;我想不出任何其他满足预期语义的好方法;如果可以的话:我全神贯注!
    • @MarcGravell 如果我们获取 protogen 生成的完整代码,然后在其上调用 Serializer.GetProto<ChangeNotification>,那么结果是否应该再次与 MS 示例中的原始 .proto 代码完全相同,或者一点也不? (如果需要,我可以提出一个新问题)。
    • @evilmandarine 它至少意味着与协议兼容; IIRC 我目前没有任何明确的方式在 GetProto/annotations 中表示 oneof;如果我们这样做会很好,作为补充
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2021-07-05
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2016-07-19
    相关资源
    最近更新 更多