如何将 protobuf-net 与通过 ASP.NET Web Reference 生成的类一起使用,同时避免代码重复?

Posted

技术标签:

【中文标题】如何将 protobuf-net 与通过 ASP.NET Web Reference 生成的类一起使用,同时避免代码重复?【英文标题】:How to use protobuf-net with classes generated through ASP.NET Web Reference while avoiding code duplication? 【发布时间】:2012-07-11 10:02:31 【问题描述】:

我有一个服务器-客户端应用程序,我可以在其中修改两个代码库。客户端通过众多的 Web 服务与服务器进行通信,我通过 Web 参考系统分享了一些在服务器端定义的类。在网络上,数据是使用 XML (SOAP) 发送的。另外,我使用XmlSerializer将一些数据保存在磁盘上。

由于性能问题不断上升,我想迁移到更复杂的序列化程序,并着眼于 Protocol Buffers 和 protobuf-net。我目前使用 protobuf-net v2 (r480, .NET 3.5)

我似乎遇到的问题是通过 Web 引用系统共享的类不保留自定义类/成员属性,如 ProtoContractProtoMember。 (但是序列化系统不会抛出通常的System.InvalidOperationException: Type is not expected, and no contract can be inferred,给我留下一个空流。是不是因为在客户端生成的类被标记为partial?)

示例,服务器端:

[ProtoContract]
public class CommentStruct

    [ProtoMember(1)] public int id;
    [ProtoMember(2)] public DateTime time;
    [ProtoMember(3)] public string comment;
    [ProtoMember(4)] public int session;

客户端(生成代码):

/// <remarks/>
[System.CodeDom.Compiler.GeneratedCodeAttribute("wsdl", "2.0.50727.1432")]
[System.SerializableAttribute()]
[System.Diagnostics.DebuggerStepThroughAttribute()]
[System.ComponentModel.DesignerCategoryAttribute("code")]
[System.Xml.Serialization.XmlTypeAttribute(Namespace="http://example.org")]
public partial class CommentStruct 
    private int idField;
    private System.DateTime timeField;
    private string commentField;
    private int sessionField;

    /// <remarks/>
    public int id 
        get 
            return this.idField;
        
        set 
            this.idField = value;
        
    
    [...]

我设法在客户端的一个额外类文件中使用ProtoPartialMember 解决了这个问题:

[ProtoContract,
ProtoPartialMember(1, "id"),
ProtoPartialMember(2, "time"),
ProtoPartialMember(3, "comment"),
ProtoPartialMember(4, "session")]
public partial class CommentStruct


这里的主要问题是:我可以用不同的方式来避免代码重复吗? 另一个是:我会错过一些 protobuf-net 的好东西,比如继承支持吗?

我找到了一些关于 protobuf-net Visual Studio 插件的信息。但正如 Marc Gravell 将其描述为“事后的想法”,我不愿意使用它。此外,我的一些合作开发者正在使用不支持加载项的 VS Express 版本。

编辑: 我的主要重复问题是必须两次指定类成员和 protobuf-net 属性 - 在服务器端的类定义和客户端的部分类属性中。

【问题讨论】:

【参考方案1】:

作为一个次要的术语,我会避免在这里使用“共享”这个词,因为“共享”通常意味着:作为一个库(可以工作)。

不抛出错误的原因与partial无关;只是 protobuf-net 很乐意使用其他库的属性......在合理范围内。例如,它将与 [XmlType] / [XmlElement(Order=n)][DataContract] / [DataMember(Order=n)] 一起使用。但是,这里从 WSDL 生成的代码不包含 Order=n,因此它不会将那些视为需要序列化的成员(它确实需要数字)。

[ProtoPartialMember(...)] 用法(正如您所提到的)是解决此问题的一种方法。它不会限制您使用其他 protobuf-net。 [ProtoInclude(...)] 仍然可以工作(因为你提到了继承)。事实上,protobuf-net 中的everything 也可以在运行时完全配置,零属性(通过RuntimeTypeMode)。重复:除了类名,我看不到任何重复。可以说,[ProtoInclude(...)] 是与[XmlInclude(...)] 重复的,但并不多。

我很高兴我顺便使用了这个短语,但我不确定“事后诸葛亮”是真的我描述插件用法的主要方式。澄清一下:protobuf-net 主要是为了适应代码优先的常见 .NET 习惯用法(可能带有属性),而不是从工具生成。它仍然有效

当然,如果可用的话,最简单的选择可能是简单地共享 .dll 或 .cs。这与 WCF / DataContractSerializer 一起工作毫不费力,但我不确定旧的“网络参考”网络服务是否非常喜欢这一点。当然,也没有插件支持将 protobuf-net 用作旧式“网络参考”(“asmx”)的一部分,因此您可能已经需要在那里更改一些内容......

【讨论】:

我的主要重复问题是必须两次指定类成员和 protobuf-net 属性 - 在服务器端的类定义和客户端的部分类属性中。直到现在我才发现这个“事后”引用的来源是三年前的:link。我很抱歉挖了这么久的线程。

以上是关于如何将 protobuf-net 与通过 ASP.NET Web Reference 生成的类一起使用,同时避免代码重复?的主要内容,如果未能解决你的问题,请参考以下文章

如何使用 RuntimeTypeModel 将 ProtoInclude 与 protobuf-net 中的类型相关联?

protobuf-net 如何序列化 DateTime?

Protobuf-net 是不是有计划包括验证?

Protobuf-Net:如何序列化 guid?

如何在 protobuf-net 中启用字符串实习?

如何通过 TCP 和 protobuf-net 接收包