3

我正在尝试更改在客户端和服务器上使用共享实体的现有 WCF net.tcp 项目中的序列化程序。我很难弄清楚 protobuf-net(V2480)这里的图表说我可以序列化私有成员,但找不到这样做的文档,没有属性可以吗?如何启用图形模式(作为参考),如此处所述

这会解决 protobuf 触发我更改的项目标志的问题吗?例如我有一堂课

 public enum FirstEnum
{ 
    First = 0,
    Second,
    Third
}
public enum AnotherEnum
{ 
    AE1 = 0,
    AE2,
    AE3
}
[Serializable()]
public class SomeClass
{
    public int SomeClassId { get; set; }
    public FirstEnum FEnum { get; set; }
    public AnotherEnum AEnum { get; set; }
    string thing;
    public string Thing
    {
        get{return thing;}
        set
        {
            if (string.IsNullOrEmpty(value))
                throw new ArgumentNullException("Thing");

            thing = value;
        }
    }

    private decimal firstAmount;
    public decimal FirstAmount 
    { 
        get{return firstAmount;} 
        set
        {
            if (value != firstAmount)
            {
                firstAmount = value;
                changedItems.Add("FirstAmount changed");
            }
        }
    }
    private decimal secondAmount;
    public decimal SecondAmount
    {
        get { return secondAmount; }
        set
        {
            if (value != secondAmount)
            {
                secondAmount = value;
                changedItems.Add("SecondAmount changed");
            }
        }
    }
    public decimal ThirdAmount { get { return SecondAmount - FirstAmount; } }
    public DateTime? SomeDate { get; set; }

    private List<string> changedItems = new List<string>();
    public List<string> ChangedItems
    {
        get { return changedItems; }
    }
public int PrivateSet { get; private set; }
    public SomeClass() { }

    public SomeClass(decimal first, decimal second)
    {
        FirstAmount = first;
        SecondAmount = second;
    }

    public void ClearChangedItems()
    {
        changedItems.Clear();
    }

当我用 (1000 items) 反序列化它时

 var model = CreateModel();
 items = (List<SomeClass>)model.Deserialize(returnStream, null, typeof(List<SomeClass>));

2012-04-06 09:14:28.1222|DEBUG|ProtobufTEsts.Form1|ProtoBuf 更改项目数:1000

使用 BinaryFormatter

System.Runtime.Serialization.Formatters.Binary.BinaryFormatter binaryFormatter = new System.Runtime.Serialization.Formatters.Binary.BinaryFormatter();
items = (List<SomeClass>)binaryFormatter.Deserialize(returnStream);

2012-04-06 09:14:28.1662|DEBUG|ProtobufTEsts.Form1|BinaryFormatter 更改项目数:0

有没有办法让 protobuf 表现得像 binaryFormatter 但保留 protobuf 的性能?

如何允许私有序列化,这失败了

 public static TypeModel CreateModel()
    {
        RuntimeTypeModel model = TypeModel.Create();
        ///var metaType = RuntimeTypeModel.Default.Add(typeof(SomeClass), false);
        model.Add(typeof(SomeClass), false)
            .Add(1, "SomeClassId")
            .Add(2, "FEnum")
            .Add(3, "AEnum")
            .Add(4, "Thing")
            .Add(5, "FirstAmount")
            .Add(6, "SecondAmount")
            .Add(7, "SomeDate")
            .Add(8, "PrivateSet");
        TypeModel compiled = model.Compile();
        return compiled;
    }
4

2 回答 2

2

啊,我现在明白了这个问题;这条线是有问题的:

TypeModel compiled = model.Compile();
return compiled;

如果您使用 Compile(),它会创建一个正式的程序集(在内存中),该程序集必须遵守程序集的通常规则,特别是:成员可访问性。这意味着它无法访问您的私人 sertter。

相反,使用:

model.CompileInPlace();
return model;

这将执行部分编译,但继续使用 DynamicMethod。这个厚脸皮的小动物可以选择通过可访问性规则进行欺骗(很像反射可以),因此它可以继续使用私有设置器。

请注意,模型也是根据需要就地编译(在更细粒度的级别),因此对 CompileInPlace 的调用不是严格必要的,但有助于提前做好所有事情。

为了完整起见,还有一个额外的 Compile(string,string) 重载,可用于在磁盘上生成单独的序列化 dll,无需在运行时进行任何元编程即可引用和使用。

于 2012-04-07T09:36:41.333 回答
0

是的 protobuf-net 可以序列化私有字段,并且没有属性。我不在电脑上,所以这可能需要调整:

var metaType = RuntimeTypeModel.Default.Add(typeof(SomeClass), false);
// for each field in a known order
metaType.Add(fieldName, someUniqueTag);

在属性驱动的用法中,还有 ImplicitFields.AllFields 会根据您的意图自动配置它,但我还没有向 MetaType 添加 ImplicitFields 辅助方法。我会把它添加到我的列表中!

注意:标签(=字段)数字对 protobuf 很重要,并且在反序列化时必须可以重现相同的数字映射。

您可能要考虑的另一个选项是(反)序列化回调,它可以让您知道它当前正在序列化/反序列化(通过 before/after 方法调用)。这可能是另一种禁用间隔的副作用的方法,例如反序列化。

于 2012-04-06T16:58:53.777 回答