我创建了一个我的对象的二进制序列化文件,因此我可以将它用作许多其他项目的数据库文件。它对同一个项目工作得很好,但是当我尝试从其他项目反序列化该文件时,它不会。出现的错误显示“找不到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();
我需要做哪些修改???请向我提供一个工作代码示例/示例。
感谢。
答案 0 :(得分:2)
BinaryFormatter
是基于类型的序列化的;它深深地将类型元数据嵌入到输出中。我打赌你已经复制项目之间的类定义 - 这对于BinaryFormatter
来说是不够的,因为那不再是Type
(类型被绑定到他们的汇编)。
在您的场景中,我觉得这里正确的做法是使用基于合同的序列化程序;例如:
XmlSerializer
,DataContractSerializer
等)JavascriptSerializer
,JSON.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名称是一些随机字符,那么我会说您正面临上述问题。您可以通过打开非文档开关来解决此问题。请参阅以下文章: