- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在尝试确定如何使用 protobuf-net(Marc Gravell 的实现)来解决这个用例。
我想到了数据协定代理和 DataContractSerializer 的概念。目标是将 version:1 数据转换为新的 B 类结构。
一个例子:
[DataContract]
public class A {
public A(){}
[DataMember]
public bool IsActive {get;set;]
[DataMember]
public int VersionNumber {
get { return 1; }
set { }
}
[DataMember]
public int TimeInSeconds {get;set;}
[DataMember]
public string Name {get;set;}
[DataMember]
public CustomObject CustomObj {get;set;} //Also a DataContract
[DataMember]
public List<ComplexThing> ComplexThings {get;set;} //Also a DataContract
...
}
[DataContract]
public class B {
public B(A a) {
this.Enabled = a.IsActive; //Property now has a different name
this.TimeInMilliseconds = a.TimeInSeconds * 1000; //Property requires math for correctness
this.Name = a.Name;
this.CustomObject2 = new CustomObject2(a.CustomObj); //Reference objects change, too
this.ComplexThings = new List<ComplexThings>();
this.ComplexThings.AddRange(a.ComplexThings);
...
}
public B(){}
[DataMember]
public bool Enabled {get;set;]
[DataMember]
public int Version {
get { return 2; }
set { }
}
[DataMember]
public double TimeInMilliseconds {get;set;}
[DataMember]
public string Name {get;set;}
[DataMember]
public CustomObject2 CustomObject {get;set;} //Also a DataContract
[DataMember]
public List<ComplexThing> ComplexThings {get;set;} //Also a DataContract
...
}
A 类是我们对象的第一次迭代,并且正在积极使用中。数据以 v1 格式存在,使用类 A 进行序列化。
在意识到我们的方法错误之后,我们创建了一个名为 B 类的新结构。A 和 B 之间有很多变化,我们觉得创建 B 比修改原始类 A 更好。
但是我们的应用程序已经存在并且类 A 正在用于序列化数据。我们已准备好将我们的更改推向世界,但我们必须首先反序列化在版本 1(使用类 A)下创建的数据并将其实例化为类 B。数据足够重要,我们不能只假设类中的默认值B 用于丢失数据,而是我们必须将数据从 A 类实例转移到 B 类。一旦我们有了 B 类实例,应用程序将再次以 B 类格式(版本 2)序列化该数据。
我们假设我们将来会对 B 类进行修改,并且我们希望能够迭代到版本 3,也许是在一个新的类“C”中。我们有两个目标:处理已经存在的数据,并为 future 的迁移准备我们的对象。
现有的“转换”属性(OnSerializing/OnSerialized、OnDeserializing/OnDeserialized 等)不提供对以前数据的访问。
在这种情况下使用 protobuf-net 的预期做法是什么?
最佳答案
对;看着他们,你确实完全改变了契约(Contract)。我知道没有任何基于契约的序列化器会因此爱上你,protobuf-net 也不异常(exception)。如果您已经有一个根节点,您可以做类似的事情(在伪代码中):
[Contract]
class Wrapper {
[Member] public A A {get;set;}
[Member] public B B {get;set;}
[Member] public C C {get;set;}
}
然后只选择 A/B/C 中非空的那个,也许在它们之间添加一些转换运算符。但是,如果您在旧数据中只有一个裸 A,这就会变得困难。我可以想到两种方法:
Version
作为初始步骤,并告诉序列化程序期望什么。例如,你可以这样做:
int version = -1;
using(var reader = new ProtoReader(inputStream)) {
while(reader.ReadFieldHeader() > 0) {
const int VERSION_FIELD_NUMBER = /* todo */;
if(reader.FieldNumber == VERSION_FIELD_NUMBER) {
version = reader.ReadInt32();
// optional short-circuit; we're not expecting 2 Version numbers
break;
} else {
reader.SkipField();
}
}
}
inputStream.Position = 0; // rewind before deserializing
现在你可以使用序列化器,告诉它什么version
它被序列化为;通过通用 Serializer.Deserialize<T>
API,或通过 Type
来自两个非通用 API 的实例(Serializer.NonGeneric.Deserialize
或 RuntimeTypeModel.Default.Deserialize
- 无论哪种方式,您都会到达同一个地方;这实际上是通用或非通用最方便的情况)。
那么你需要一些 A
之间的转换代码/B
/C
- 通过您自己的自定义运算符/方法,或通过自动映射器之类的东西。
如果你不想要任何 ProtoReader
代码踢,你也可以这样做:
[DataContract]
class VersionStub {
[DataMember(Order=VERSION_FIELD_NUMBER)]
public int Version {get;set;}
}
然后运行Deserialize<VersionStub>
, 这将使您可以访问 Version
,然后您可以使用它来执行特定于类型的反序列化;这里的主要区别在于 ProtoReader
代码允许您在获得版本号后立即短路。
关于c# - protobuf-net、版本控制和代理类型的最佳实践,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14491140/
我试图了解是否有可能采用构成另一个 protobuf 一部分的序列化 protobuf 并将它们合并在一起而不必反序列化第一个 protobuf。 例如,给定一个 protobuf 包装器: synt
正如我最近发现的,我们可以使用两个类 ProtoBuf.Serializer 和 ProtoBuf.Meta.TypeModel 在 protobuf-net 中序列化/反序列化。例如,假设我们有一个
我正在尝试使用 protobuf 序列化我的下面的类,但它因“对象引用”错误而失败。更多详情如下。通过查看错误详细信息,您知道会出现什么问题吗?注意:我的用户对象太大了,它有太多的子对象和属性。所以不
我想识别要反序列化的文件是否是protobuf序列化的。这是因为我想提供不止一种选项来为用户序列化和反序列化文件。我正在使用 protobuf.net 序列化为 protobuf 格式。 最佳答案 不
我已经使用位于 https://protogen.marcgravell.com/ 的工具构建了我的 C# 类来自 https://developers.google.com/transit/gtfs
有一个通过 UDP 接受消息的 Go 服务器。使用这种设计,它只能扫描一种特定类型的实体,world.Entity . for { buf := make([]byte, 10
比如我想序列化和反序列化System.Drawing.Font这是不可变的,不能更改以适应 protobuf-net 约定。一般来说,是否可以在 protobuf-net 中编写某种“自定义”序列化程
我开始用 protobuf 2.2.0 构建一个应用程序,因为它是最新的。现在我正在考虑升级到最新的 protobuf 2.4.0a。 如果我这样做,对于同一架构,一个版本的应用程序生成的消息是否仍然
在我从 BinaryFormatter 切换到 protobuf-net 的过程中, 我在序列化集合时观察到了差异。 在下面的代码示例中,反序列化(protobuf-net v2r470)返回 如果在
知道正在发送的 protobuf 消息类型的 API 是什么? 例如,我使用以下方法获取 SendNameMessage 对象。 SendNameMessage sendNameObj = Seria
我在我们的一个项目中使用 protobuf-net 来序列化/反序列化一大组同类对象。它运行良好,速度非常快。不过只有一个问题。反序列化时是否可以使用 linq(或任何其他机制)指定过滤条件,以便加载
我正在尝试使用 protobuf-net 序列化一些对象,但不幸的是他们自由地使用了 DateTimeOffset , protobuf-net 尚不支持。这导致了很多: No serializer
我在 ionic2 项目中使用 protobuf.js。我有一个有效的 .proto 文件,我首先将其转换为静态 javascript 文件: pbjs -t static databaseapi.p
我通过 vcpkg vcpkg install protobuf:x64-windows 安装了 protobuf .显然它安装了最新版本(3.6.1)。对于我需要版本<=3.5.1的项目。有没有办法
我有以下类(class):- [Serializable] [DataContract(Name = "StateValueWrapper")] public class StateValueWrap
protobuf net 似乎不支持列表/数组的 AsReference 以及列表/数组内对象的 AsReference。这会在最终的 v2 中得到支持吗? [ProtoMember(1, AsRef
我正在使用 protobuf-net 来序列化和反序列化我的消息。我的消息还包含可以为空的字符串。但是,当我在另一侧反序列化它们时,我得到空字符串 ("")。 根据谷歌文档,空字符串中字符串类型的默认
我已经阅读了有关继承的各种帖子,并且 Protocol Buffer 不支持继承。我不想继承 Protocol Buffers 消息,而是继承,这样我就可以轻松处理我的所有 Protocol Buff
我知道带有 protobuf.net 的列表不支持 AsReference,因此我尝试了解决此限制的方法。我创建了一个名为 SuperList 的自定义列表,其中包含包装在 SuperListItem
我正在尝试使用 ProtoMember 中的 AsReference 选项进行递归引用。如果我使用公共(public)构造函数创建 AnOwner 然后序列化/反序列化,AnOwner.Data 变为
我是一名优秀的程序员,十分优秀!