4

基于此处找到的代码:protobuf 和 List<object> - 如何序列化/反序列化?我创建了一个具有 ProtoObject 值类型的通用“ProtoDictionary”。

这是我的 ProtoDictionary 代码:

public class ProtoDictionary<TKey> : Dictionary<TKey, ProtoObject>
{
    public void Add(TKey key, string value)
    {
        base.Add(key, new ProtoObject<<string>(value));
    }

    public void Add(TKey key, List<string> value)
    {
        base.Add(key, new ProtoObject<List<string>>(value));
    }

    public void Add(TKey key, List<UrlStatus> value)
    {
        base.Add(key, new ProtoObject<List<UrlStatus>>(value));
    }

    public void Add(TKey key, Dictionary<string, string> value)
    {
        base.Add(key, new ProtoObject<Dictionary<string, string>>(value));
    }

    public void Add(TKey key, Dictionary<string, int> value)
    {
        base.Add(key, new ProtoObject<Dictionary<string, int>>(value));
    }

    public void Add(TKey key, List<TrafficEntry> value)
    {
        base.Add(key, new ProtoObject<List<TrafficEntry>>(value));
    }

    public ProtoDictionary()
    {
        // Do nothing
    }

    // NOTE: For whatever reason, this class will not correctly deserialize without this method, even though
    // the base class, Dictionary, has the SerializableAttribute. It's protected so only the framework can access it.
    protected ProtoDictionary(SerializationInfo info, StreamingContext context) : base(info, context) 
    {

    }
}

对于原型对象:

[ProtoContract]
[ProtoInclude(1, typeof(ProtoObject<string>))]
[ProtoInclude(2, typeof(ProtoObject<int>))]
[ProtoInclude(3, typeof(ProtoObject<List<string>>))]
[ProtoInclude(4, typeof(ProtoObject<Dictionary<string, string>>))]
[ProtoInclude(5, typeof(ProtoObject<List<TrafficEntry>>))]
[ProtoInclude(6, typeof(ProtoObject<Dictionary<string, int>>))]
[ProtoInclude(7, typeof(ProtoObject<bool>))]
[ProtoInclude(8, typeof(ProtoObject<double>))]
[ProtoInclude(9, typeof(ProtoObject<decimal>))]
[ProtoInclude(10, typeof(ProtoObject<float>))]
[ProtoInclude(11, typeof(ProtoObject<long>))]
[ProtoInclude(12, typeof(ProtoObject<SerializableException>))]
[ProtoInclude(13, typeof(ProtoObject<List<UrlStatus>>))]
[Serializable]
public abstract class ProtoObject
{
    public static ProtoObject<T> Create<T>(T value)
    {
        return new ProtoObject<T>(value);
    }

    public object Value
    {
        get { return ValueImpl; }
        set { ValueImpl = value; }
    }

    protected abstract object ValueImpl { get; set; }

    protected ProtoObject()
    {

    }
}

[ProtoContract]
[Serializable]
public sealed class ProtoObject<T> : ProtoObject
{
    public ProtoObject()
    {

    }

    public ProtoObject(T value)
    { 
        Value = value; 
    }

    [ProtoMember(1)]
    public new T Value { get; set; }

    protected override object ValueImpl
    {
        get { return Value; }
        set { Value = (T)value; }
    }

    public override string ToString()
    {
        return Value.ToString();
    }
}

问题是,当我尝试使用以下代码从 SQL 反序列化 ProtoDictionary 时:

public T Deserialize<T>(IDataReader reader, string columnName)
{
    MemoryStream stream = new MemoryStream();
    byte[] buffer = new byte[256];          
    long startIndex = 0;
    long bytesRead = reader.GetBytes(reader.GetOrdinal(columnName), startIndex, buffer, 0, buffer.Length);

    while(bytesRead == buffer.Length)
    {
        stream.Write(buffer, 0, (int)bytesRead);                
        startIndex += bytesRead;
        bytesRead = reader.GetBytes(reader.GetOrdinal(columnName), startIndex, buffer, 0, buffer.Length);
    }

    stream.Write(buffer, 0, (int)bytesRead);
    stream.Seek(0, SeekOrigin.Begin);

    return (T)Utilities.Deserialize<T>(stream);
}

我收到错误“InvalidOperationException:无法创建抽象类的实例”。

我的 StackTrace 如下:

   在 ctorWrapper()
   在 ProtoBuf.ObjectFactory`1.Create() 在 c:\protobuf-net_fixed\trunk\protobuf-net\ObjectFactory.cs:line 82
   在 ProtoBuf.Serializer`1.Deserialize[TCreation](T& instance, SerializationContext context) in c:\protobuf-net_fixed\trunk\protobuf-net\SerializerT.cs:line 568
   在 ProtoBuf.Property.PropertyMessageString`4.DeserializeImpl(TSource source, SerializationContext context) 在 c:\protobuf-net_fixed\trunk\protobuf-net\Property\PropertyMessageString.cs:line 53
   在 ProtoBuf.Property.PropertyPairString`3.DeserializeImpl(TSource source, SerializationContext context) 在 c:\protobuf-net_fixed\trunk\protobuf-net\Property\PropertyPairString.cs:line 53
   在 ProtoBuf.Property.PropertyList`3.DeserializeImpl(TSource source, SerializationContext context, Boolean canSetValue) in c:\protobuf-net_fixed\trunk\protobuf-net\Property\PropertyList.cs:line 64
   在 ProtoBuf.Property.PropertyList`3.Deserialize(TSource source, SerializationContext context) 在 c:\protobuf-net_fixed\trunk\protobuf-net\Property\PropertyList.cs:line 52
   在 ProtoBuf.Serializer`1.Deserialize[TCreation](T& instance, SerializationContext context) in c:\protobuf-net_fixed\trunk\protobuf-net\SerializerT.cs:line 568
   在 ProtoBuf.Serializer`1.DeserializeChecked[TCreation](T& instance, SerializationContext source) in c:\protobuf-net_fixed\trunk\protobuf-net\SerializerT.cs:line 400
   在 ProtoBuf.SerializerSimpleProxy`1.Deserialize(TValue& value, SerializationContext source) 在 c:\protobuf-net_fixed\trunk\protobuf-net\SerializerProxy.cs:line 100
   在 c:\protobuf-net_fixed\trunk\protobuf-net\Serializer.cs:line 302 中的 ProtoBuf.Serializer.Deserialize[T](SerializationContext source)
   在 ProtoBuf.Serializer.Deserialize[T](Stream source) in c:\protobuf-net_fixed\trunk\protobuf-net\Serializer.cs:line 289
   在 C:\QA\trunk\TestFramework\Core\Utilities.cs:line 312 中的 Demand.TestFramework.Core.Utilities.Deserialize[T](MemoryStream 流)
   在 Demand.TestFramework.Core.Reports.CrawlerReport.Deserialize[T](IDataReader reader, String columnName) 在 C:\QA\trunk\TestFramework\Core\Reports\CrawlerReport.cs:line 145
   在 C:\QA\trunk\TestFramework\Core\Reports\CrawlerReport.cs:line 70 中的 Demand.TestFramework.Core.Reports.CrawlerReport.FormatSuite(Int32 parentSuiteId, Guid runId)
   在 C:\QA\trunk\TestFramework\Core\Reports\CrawlerReport.cs:line 150 中的 Demand.TestFramework.Core.Reports.CrawlerReport.Format(Guid runId)
   在 C:\QA\trunk\TestFramework\ServiceLauncher\Form1.cs:line 24 中的 ServiceLauncher.Form1.btnStart_Click(Object sender, EventArgs e)
   在 System.Windows.Forms.Control.OnClick(EventArgs e)
   在 System.Windows.Forms.Button.OnClick(EventArgs e)
   在 System.Windows.Forms.Button.OnMouseUp(MouseEventArgs 事件)
   在 System.Windows.Forms.Control.WmMouseUp(消息和 m,MouseButtons 按钮,Int32 点击)
   在 System.Windows.Forms.Control.WndProc(消息和 m)
   在 System.Windows.Forms.ButtonBase.WndProc(消息和 m)
   在 System.Windows.Forms.Button.WndProc(消息和 m)
   在 System.Windows.Forms.Control.ControlNativeWindow.OnMessage(消息& m)
   在 System.Windows.Forms.Control.ControlNativeWindow.WndProc(消息和 m)
   在 System.Windows.Forms.NativeWindow.DebuggableCallback(IntPtr hWnd,Int32 msg,IntPtr wparam,IntPtr lparam)
   在 System.Windows.Forms.UnsafeNativeMethods.DispatchMessageW(味精和味精)
   在 System.Windows.Forms.Application.ComponentManager.System.Windows.Forms.UnsafeNativeMethods.IMsoComponentManager.FPushMessageLoop(IntPtr dwComponentID,Int32 原因,Int32 pvLoopData)
   在 System.Windows.Forms.Application.ThreadContext.RunMessageLoopInner(Int32 原因,ApplicationContext 上下文)
   在 System.Windows.Forms.Application.ThreadContext.RunMessageLoop(Int32 原因,ApplicationContext 上下文)
   在 System.Windows.Forms.Application.Run(窗体 mainForm)
   在 C:\QA\trunk\TestFramework\ServiceLauncher\Program.cs:line 16 中的 ServiceLauncher.Program.Main()
   在 System.AppDomain._nExecuteAssembly(RuntimeAssembly 程序集,字符串 [] 参数)
   在 System.AppDomain.ExecuteAssembly(字符串 assemblyFile,证据 assemblySecurity,String [] args)
   在 Microsoft.VisualStudio.HostingProcess.HostProc.RunUsersAssembly()
   在 System.Threading.ThreadHelper.ThreadStart_Context(对象状态)
   在 System.Threading.ExecutionContext.Run(ExecutionContext executionContext,ContextCallback 回调,对象状态,布尔 ignoreSyncCtx)
   在 System.Threading.ExecutionContext.Run(ExecutionContext executionContext,ContextCallback 回调,对象状态)
   在 System.Threading.ThreadHelper.ThreadStart()

我不确定我做错了什么。任何帮助将不胜感激。

谢谢,丹

4

2 回答 2

1

这可能只是“v1”中的一个限制。我已将此添加为对“v2”的测试,并且它通过了(我必须发明UrlStatusTrafficEntry让它编译):

    public enum UrlStatus { A,B }
    public enum TrafficEntry { A }
    [ProtoContract]
    public class SerializableException { }

    [Test]
    public void TestBasicRoundTrip()
    {
        var item = new ProtoDictionary<string>();
        item.Add("abc", "def");
        item.Add("ghi", new List<UrlStatus> {UrlStatus.A, UrlStatus.B});

        var clone = Serializer.DeepClone(item);
        Assert.AreEqual(2, clone.Keys.Count);
        object o = clone["abc"];
        Assert.AreEqual("def", clone["abc"].Value);
        var list = (IList<UrlStatus>)clone["ghi"].Value;
        Assert.AreEqual(2, list.Count);
        Assert.AreEqual(UrlStatus.A, list[0]);
        Assert.AreEqual(UrlStatus.B, list[1]);
    }

使用“v1”,也许根本就做不到abstract?(解决方法,不修复)

还; 应该不需要SerializationInfoctor;protobuf-net 不使用它(尽管您可以通过提供此方法并调用 into来实现protobuf-net 内部)BinaryFormatterMerge

于 2010-07-16T09:11:08.407 回答
0

是否有理由需要将 ProtoObject 和 ProtoObject<T> 分开?protobuf-net 似乎在 ObjectFactory 中创建了一个 T 类型的对象(以迂回的方式),我假设 T 将是抽象的 ProtoObject。

抱歉,我没有更多帮助 - 目前我没有太多时间进一步挖掘。

于 2010-07-16T02:33:00.250 回答