2012-08-28 265 views
2

我有一個類。我想序列化和序列化包含該類的對象的列表。我想使用二進制格式化程序。 這是我的課:列表序列化和二進制格式化的反序列化

[Serializable] 
public class Section :ISerializable 
{ 
    private double ClearCover; 
    private string SectionName; 

    private double Diameter; 

    public double Height; 
    private double Width; 

    private double InternalDiameter; 
    private double ExternalDiameter; 

    private double InternalHeight; 
    private double ExternalHeight; 
    private double InternalWidth; 
    private double ExternalWidth; 

    private double WebWidth; 
    private double FlangeWidth; 
    private double FlangeThickness; 



    public double clearCover 
    { 
     get { return ClearCover; } 
     set { ClearCover = value; } 
    } 
    public string sectionName 
    { 
     get{return SectionName;} 
     set{SectionName=value;} 
    } 

    public double diameter 
    { 
     get { return Diameter; } 
     set { Diameter = value; } 
    } 

    public double height 
    { 
     set { Height = value; } 
     get { return Height; } 
    } 
    public double width 
    { 
     set { Width = value; } 
     get { return Width; } 
    } 

    public double internalDiameter 
    { 
     set { InternalDiameter = value; } 
     get { return InternalDiameter; } 
    } 
    public double externalDiameter 
    { 
     set { ExternalDiameter = value; } 
     get { return ExternalDiameter; } 
    } 

    public double internalHeight 
    { 
     set { InternalHeight = value; } 
     get { return InternalHeight; } 
    } 
    public double externalHeight 
    { 
     set { ExternalHeight = value; } 
     get { return ExternalHeight; } 
    } 
    public double internalWidth 
    { 
     set { InternalWidth = value; } 
     get { return InternalWidth; } 
    } 
    public double externalWidth 
    { 
     set { ExternalWidth = value; } 
     get { return ExternalWidth; } 
    } 

    public double flangeWidth 
    { 
     set { FlangeWidth = value; } 
     get { return FlangeWidth; } 
    } 
    public double flangeThickness 
    { 
     set { FlangeThickness = value; } 
     get { return FlangeThickness; } 
    } 
    public double webWidth 
    { 
     set { WebWidth = value; } 
     get { return WebWidth; } 
    } 




    public Section() { } 

    protected Section(SerializationInfo info, StreamingContext context) 
    { 
     sectionName = info.GetString("Section Name"); 
     clearCover = info.GetDouble("Clear Cover"); 

     diameter = info.GetDouble("Diameter"); 

     //internalDiameter = info.GetDouble("Internal Diameter"); 
     //externalDiameter = info.GetDouble("External Diameter"); 


     height = info.GetDouble("Height"); 
     width = info.GetDouble("Width"); 



     internalHeight = info.GetDouble("Internal Height"); 
     externalHeight = info.GetDouble("External Height"); 
     internalWidth = info.GetDouble("Internal Width"); 
     externalWidth = info.GetDouble("External Width"); 

     flangeWidth = info.GetDouble("Flange Width"); 
     flangeThickness = info.GetDouble("Flange Thickness"); 
     webWidth = info.GetDouble("Web Width"); 



    } 

    public virtual void GetObjectData(SerializationInfo info, StreamingContext context) 
    { 
     info.AddValue("Section Name", sectionName); 
     info.AddValue("Clear Cover", clearCover); 

     info.AddValue("Diameter",diameter); 

     //info.AddValue("Internal Diameter", internalDiameter); 
     //info.AddValue("External Diameter", externalDiameter); 

     info.AddValue("Height",height); 
     info.AddValue("Width",width); 

     info.AddValue("Internal Height",internalHeight); 
     info.AddValue("External Height",externalHeight); 
     info.AddValue("Internal Width",internalWidth); 
     info.AddValue("External Width",externalWidth); 

     info.AddValue("Flange Width",flangeWidth); 
     info.AddValue("Flange Thickness", flangeThickness); 
     info.AddValue("Web Width", webWidth); 


    } 


} 

我沒有序列化任何問題。但是對於反序列化,它可能只是反序列化第一個對象,而不是全部。我該怎麼做,請幫助我!

回答

2

從從OP一個單獨的電子郵件,得到了如下的方法(我已經整理他們一點,不要太多):

static void Serialize(object obj, string filename) 
{ 
    using (FileStream streamOut = new FileStream(filename, FileMode.Append)) 
    { 
     BinaryFormatter formatter = new BinaryFormatter(); 
     formatter.Serialize(streamOut, obj); 
    } 
} 


public static List<Section> DeserializeList(string filename) 
{ 
    using (FileStream streamIn = File.OpenRead(filename)) 
    { 
     BinaryFormatter formatter = new BinaryFormatter(); 
     return (List<Section>) formatter.Deserialize(streamIn); 
    } 
} 

這裏的關鍵發現是,它是假設BinaryFormatter是可附加格式,並依次串行化兩個對象是相同的序列化2項在同一時間(列表),例如:

if(File.Exists("foo.bin")) File.Delete("foo.bin"); // start afresh 
Serialize(new Section { Diameter = 1.2, ClearCover = 3.4 }, "foo.bin"); 
Serialize(new Section { Diameter = 5.6, ClearCover = 7.8 }, "foo.bin"); 

var clone = DeserializeList("foo.bin"); 

然而,簡單地並非如此。反序列化這樣的項目,我們將不得不做這樣的事情:

public static List<Section> DeserializeList(string filename) 
{ 
    using (FileStream streamIn = File.OpenRead(filename)) 
    { 
     List<Section> list = new List<Section>(); 
     BinaryFormatter formatter = new BinaryFormatter(); 
     while(streamIn.Position != streamIn.Length) 
     { 
      list.Add((Section) formatter.Deserialize(streamIn)); 
     } 
     return list; 
    } 
} 

它有點讓人頭疼的 - 尤其是因爲它不能被應用到所有的流(.Length甚至.Position尚未普及)。

以上應該工作,但我強烈建議使用可附加格式,如protobuf。事實上,protobuf網,使這種情況下毫不費力,無論是在模型方面:

[ProtoContract] 
public class Section 
{ 
    [ProtoMember(1)] public double ClearCover { get; set; } 
    [ProtoMember(2)] public string SectionName { get; set; } 
    [ProtoMember(3)] public double Diameter { get; set; } 
    [ProtoMember(4)] public double Height { get; set; } 
    [ProtoMember(5)] public double Width { get; set; } 
    [ProtoMember(6)] public double InternalDiameter { get; set; } 
    [ProtoMember(7)] public double ExternalDiameter { get; set; } 
    [ProtoMember(8)] public double InternalHeight { get; set; } 
    [ProtoMember(9)] public double ExternalHeight { get; set; } 
    [ProtoMember(10)] public double InternalWidth { get; set; } 
    [ProtoMember(11)] public double ExternalWidth { get; set; } 
    [ProtoMember(12)] public double FlangeWidth { get; set; } 
    [ProtoMember(13)] public double FlangeThickness { get; set; } 
    [ProtoMember(14)] public double WebWidth { get; set; } 
} 

和序列化/反序列化:

static void Main() 
{ 
    if (File.Exists("foo.bin")) File.Delete("foo.bin"); // start afresh 

    Serialize(new Section { Diameter = 1.2, ClearCover = 3.4 }, "foo.bin"); 
    Serialize(new Section { Diameter = 5.6, ClearCover = 7.8 }, "foo.bin"); 

    var clone = DeserializeList("foo.bin"); 
} 
static void Serialize(object obj, string filename) 
{ 
    using (FileStream streamOut = new FileStream(filename, FileMode.Append)) 
    { 
     Serializer.NonGeneric.SerializeWithLengthPrefix(
      streamOut, obj, PrefixStyle.Base128, Serializer.ListItemTag); 
    } 
} 


public static List<Section> DeserializeList(string filename) 
{ 
    using (FileStream streamIn = File.OpenRead(filename)) 
    { 
     return Serializer.DeserializeItems<Section>(
      streamIn, PrefixStyle.Base128, Serializer.ListItemTag).ToList(); 
    } 
} 

最後,BinaryFormatter數據的兩行是750個字節;與protobuf網:40字節。

+0

非常感謝:) –

+0

對於第一部分(使用二進制格式化程序),它給了我這個錯誤無法投射'System.Collections.Generic.List'1 [KSU.Section]類型的對象來鍵入'KSU 。部分'。這沒有意義。根據你寫的代碼,它不會把任何東西投給Section Type對象 –

+0

我只是自己犯了一個錯誤。它的工作原理,謝謝:) –