gpt4 book ai didi

c# - 在 Protobuf-net 中,我如何传递一个类型对象数组,其中包含不同类型的对象,提前知道潜在类型的集合

转载 作者:行者123 更新时间:2023-11-30 15:48:22 34 4
gpt4 key购买 nike

我正在尝试将使用 XmlSerializer 的现有代码迁移到 protobuf-net,因为它提供了更高的性能,但是我在这个特定案例中遇到了问题。

我有一个对象 [],其中包含将要发送到远程主机的参数(某种自定义迷你 rpc 工具)。我知道这些参数可能来自哪些类型,但我无法提前知道它们将以何种顺序发送。我有三个约束。首先是我在 Compact Framework 中运行,所以我需要在那里工作的东西。其次,正如我提到的,性能是一个大问题(在序列化方面),所以我宁愿尽可能避免在那里使用大量反射。最重要的是我关心发送这些参数的顺序。使用 XmlSerializer 只需添加 XmlInclude 即可轻松实现,但据我所知,在 Protobuf-net 中没有任何等效字段。那么,有没有办法做到这一点?这是一个简化的示例。

    [Serializable]
[XmlInclude(typeof(MyType1)),
XmlInclude(typeof(MyType2)),
XmlInclude(typeof(MyType3))
public class Message()
{
public object[] parameters;

public Message(object[] parms)
{
parameters = parms;
}
}

Message m = new Message(new object[] {MyType1(), 33, "test",
new MyType3(), new MyType3()});
MemoryStream ms = new MemoryStream();
XmlSerializer xml = new XmlSerializer(typeof(Message));
xml.Serialize(ms,xml);

这仅适用于 XmlSerializer,但如果我尝试将其转换为 protobuf-net,我将收到“没有对象的默认编码”消息。

我想出的最好办法是使用泛型和 [ProtoInclude],如本 example 所示.因为我可以在数组中有不同的对象类型,所以这不太合适。我为每个潜在类型添加了一个通用列表,并添加了一个带有 [ProtoIgnore] 和类型 object[] 的属性来添加和获取它们。添加它们时我必须使用反射(以知道将每个项目放在哪个数组中)这是不可取的而且我仍然无法保留顺序因为我只是一个接一个地提取每个列表中的所有项目并将它们放入一个属性 get 上的新对象 [] 数组。

请问有什么办法可以实现吗?


我尝试了下面 Marc 的建议,但无法正常工作。我想我可能误解了什么。

使用您编写的代码。我想我应该使用 MessageParam Create 来生成 MessageParam 对象以添加到列表中。所以基本上我像这样向 Message 添加了一个构造函数:

public Message(object[] parms)
{
foreach (object o in parms)
{
parameters.Add(MessageParam.Create(o));
}
}

但是,如果我这样做,我将得到“在序列化过程中发现意外类型;类型必须包含在 ProtoIncludeAttribute 中;发现 MessageParam`1 作为 MessageParam 传递”,因为我假设序列化程序需要非通用版本。我误解了你的建议吗?如果是这样,正确的做法是什么?

最佳答案

object 会出问题。我会尝试更像:

[ProtoContract]
class Message
{
private readonly List<MessageParam> parameters = new List<MessageParam>();
[ProtoMember(1)]
public List<MessageParam> Parameters { get { return parameters; } }
}
[ProtoContract]
[ProtoInclude(3, typeof(MessageParam<int>))]
[ProtoInclude(4, typeof(MessageParam<float>))]
[ProtoInclude(5, typeof(MessageParam<DateTime>))]
//...known types...
abstract class MessageParam {
public abstract object UntypedValue { get; set; }
public static MessageParam<T> Create<T>(T value) {
return new MessageParam<T> { Value = value };
}
public static MessageParam CreateDynamic(object value)
{
Type type = value.GetType();
switch (Type.GetTypeCode(value.GetType()))
{
// special cases
case TypeCode.Int32: return Create((int)value);
case TypeCode.Single: return Create((float)value);
case TypeCode.DateTime: return Create((DateTime)value);
// fallback in case we forget to add one, or it isn't a TypeCode
default:
MessageParam param = (MessageParam)Activator.CreateInstance(
typeof(MessageParam<>).MakeGenericType(type));
param.UntypedValue = value;
return param;
}
}
}
[ProtoContract]
sealed class MessageParam<T> : MessageParam
{
[ProtoMember(1)]
public T Value { get; set; }
public override object UntypedValue
{
get { return Value; }
set { Value = (T)value; }
}
}

请注意,未发布的“v2”代码提供了更多在运行时定义关系的能力,而不是通过属性(这在这里非常有限)。

关于c# - 在 Protobuf-net 中,我如何传递一个类型对象数组,其中包含不同类型的对象,提前知道潜在类型的集合,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2678249/

34 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com