.NET中的反序列化问题

时间:2011-09-08 05:34:31

标签: c# .net serialization binary deserialization

我创建了一个我的对象的二进制序列化文件,因此我可以将它用作许多其他项目的数据库文件。它对同一个项目工作得很好,但是当我尝试从其他项目反序列化该文件时,它不会。出现的错误显示“找不到xxxx.xxxx程序集”。那么,我应该如何序列化对象以使其独立于程序集?

这是我的代码:

            // Binary formatter
            IFormatter formatter = new BinaryFormatter();
            Stream stream = new FileStream("BinarySerialization.bin",
                                     FileMode.Create,
                                     FileAccess.Write, FileShare.None);
            formatter.Serialize(stream, words);
            stream.Close();

我需要做哪些修改???请向我提供一个工作代码示例/示例。

感谢。

3 个答案:

答案 0 :(得分:2)

BinaryFormatter是基于类型的序列化的;它深深地将类型元数据嵌入到输出中。我打赌你已经复制项目之间的类定义 - 这对于BinaryFormatter来说是不够的,因为那不再是Type(类型被绑定到他们的汇编)。

在您的场景中,我觉得这里正确的做法是使用基于合同的序列化程序;例如:

  • xml(XmlSerializerDataContractSerializer等)
  • json(JavascriptSerializer,JSON.net等)
  • 二进制(protobuf-net等)

在您的方案中会有完全,并且还有更好的版本容差(BinaryFormatter版本非常脆弱)

你提到“XML在这里不安全,因为我不希望用户知道主数据库文件的内容。” - 在这种情况下,protobuf-net具有人类不可读的“优势”,但请注意:这些都没有,BinaryFormatter 加密;如果我想,我可以获取内容,如果我真的,真的想。如果需要强大的安全性,请使用正确的加密。在这种情况下,您的代码变为(除了可能还有一些标记属性):

using(var stream = new FileStream("BinarySerialization.bin", FileMode.Create,
                                 FileAccess.Write, FileShare.None))
{
    Serializer.Serialize(stream, words);
}

编辑以显示(按评论)如何序列化Dictionary<string, List<Word>>其中Word是一个包含2个字符串成员的类(这里的大多数代码只是用于显示完整示例的管道):

using System;
using System.Collections.Generic;
using System.IO;
using ProtoBuf;
[ProtoContract]
public class Word {
    [ProtoMember(1)]
    public string Foo { get; set; }    
    [ProtoMember(2)]
    public string Bar { get; set; }
}
static class Program {
    public static void Main() {
        var data = new Dictionary<string, List<Word>>{
            {"abc", new List<Word> {
                new Word { Foo = "def", Bar = "ghi"},
                new Word { Foo = "jkl", Bar = "mno"}
            }},
            {"pqr", new List<Word> {
                new Word {Foo = "stu", Bar = "vwx"}
            }}
        };
        using(var file = File.Create("my.bin")) {
            Serializer.Serialize(file, data);
        }
        Dictionary<string, List<Word>> clone;
        using(var file = File.OpenRead("my.bin")) {
            clone = Serializer.Deserialize<
                 Dictionary<string, List<Word>>>(file);
        }
        foreach(var pair in clone) {
            Console.WriteLine(pair.Key);
            foreach(var word in pair.Value){
                Console.WriteLine("\t{0} | {1}", word.Foo, word.Bar);
            }
        }    
    }
}

答案 1 :(得分:1)

我会将所有必须序列化的模型放入单独的程序集中。 然后,您可以在需要反序列化模型的任何位置引用此程序集。

如果不需要某种生成器,则会根据某些模式重新创建模型(与WCF的实用程序相同)或使用XML等纯格式来保存数据。

答案 2 :(得分:1)

.NET中的序列化机制创建了一个类型的帮助器dll,用于在运行时序列化和反序列化数据。首先,它会吐出一个被编译的代码文件,然后加载helper dll来进行序列化和反序列化。

如果由于某种原因在创建帮助器.dll时发生了某些事情 - 假设编译错误 - 则运行时将找不到此dll。

如果您的案例中的dll名称是一些随机字符,那么我会说您正面临上述问题。您可以通过打开非文档开关来解决此问题。请参阅以下文章:

HOW TO: Debug into a .NET XmlSerializer Generated Assembly