C# socket传输自定义对象

来源:互联网 发布:实验二网络应用层协议 编辑:程序博客网 时间:2024/05/16 07:06

之前研究过自定义对象的socket传输,当时认识太浅了,实在是无法理解,现在重新研究了一次,调试通过,仅作为记录笔记。

C#中,目前我所知道的序列化与反序列化有两种,当然有更多,但是我还没有去认识。

这两种序列化方式分为:

  1. System.Runtime.Serialization.Formatters.Binary空间下的BinaryFormatter。
  2. System.Runtime.Serialization空间下的DataContractSerializer。

      第一种序列化就我目前认知来看,不能作为不同工程之前的序列化操作,也就是说不同命名空间下的程序序列化出来的信息不能共享,我尝试着在一个工程中序列化数据,然后使用socket传输给第二个工程来反序列化,结果报错,报错的大致意思就是说我序列化的内容在我反序列化的时候不能找到精确的类型来匹配,想想也是,序列化和反序列化实在两个不同的工程中,命名空间和类的定义都不一样,反正没有深究。

      第二种序列化不用担心这个问题,因为DataContractSerializer在初始化的时候必须要传递一个参数来实现构造,这个参数就是我将做的序列化操作对应的类的类型是什么,这样的话,我根本就不用担心类型匹配的问题。但是有一点要注意的是,DataContractSerializer没有在默认的这个空间中,需要引用System.Runtime.Serialization.dll才可以使用这个函数。

     第一种方式的序列化和反序列化:

[c-sharp] view plaincopyprint?
  1. public Stream serializeStream_sameProject(object param)
  2.     BinaryFormatter formmatter = new BinaryFormatter();
  3.     MemoryStream stream = new MemoryStream();
  4.  
  5.     formmatter.Serialize(stream, param); 
  6.  
  7.     return stream; 
[c-sharp] view plaincopyprint?
  1. public Stream serializeStream_sameProject(object param) 
  2.     BinaryFormatter formmatter = new BinaryFormatter(); 
  3.     MemoryStream stream = new MemoryStream(); 
  4.  
  5.     formmatter.Serialize(stream, param); 
  6.  
  7.     return stream; 

[c-sharp] view plaincopyprint?
  1. public object deserializeStream_sameProject(Stream stream)
  2.     BinaryFormatter formatter = new BinaryFormatter();
  3.  
  4.     stream.Seek(0, SeekOrigin.Begin); 
  5.  
  6.     object obj = formatter.Deserialize(stream);
  7.  
  8.     return obj; 
[c-sharp] view plaincopyprint?
  1. public object deserializeStream_sameProject(Stream stream) 
  2.     BinaryFormatter formatter = new BinaryFormatter(); 
  3.  
  4.     stream.Seek(0, SeekOrigin.Begin); 
  5.  
  6.     object obj = formatter.Deserialize(stream); 
  7.  
  8.     return obj; 

   第二种方式的序列化和反序列化:

[c-sharp] view plaincopyprint?
  1. public Stream serializeStream(object param)
  2.     DataContractSerializer serializer = new DataContractSerializer(param.GetType()); 
  3.     MemoryStream stream = new MemoryStream();
  4.  
  5.     serializer.WriteObject(stream, param); 
  6.  
  7.     return stream; 
[c-sharp] view plaincopyprint?
  1. public Stream serializeStream(object param) 
  2.     DataContractSerializer serializer = new DataContractSerializer(param.GetType()); 
  3.     MemoryStream stream = new MemoryStream(); 
  4.  
  5.     serializer.WriteObject(stream, param); 
  6.  
  7.     return stream; 

[c-sharp] view plaincopyprint?
  1. public object deserializeStream(Stream stream,Type paramType)
  2.     DataContractSerializer serializer = new DataContractSerializer(paramType); 
  3.  
  4.     stream.Seek(0, SeekOrigin.Begin); 
  5.  
  6.     object obj = serializer.ReadObject(stream);
  7.  
  8.     return obj; 
[c-sharp] view plaincopyprint?
  1. public object deserializeStream(Stream stream,Type paramType) 
  2.     DataContractSerializer serializer = new DataContractSerializer(paramType); 
  3.  
  4.     stream.Seek(0, SeekOrigin.Begin); 
  5.  
  6.     object obj = serializer.ReadObject(stream); 
  7.  
  8.     return obj; 

关于socket传输对象的办法,使用第二种序列化,将对象转换为byte数组,使用socket传输过去。

接收方接收到数组之后,又将数组反序列化回对象。

传输过程中的对象必须传输方和接收方都可以识别。

序列化的类必须以[Serializable]标明。


最近无聊在写个传输的东西,发现个问题,现在补充一下,2012-01-30

如果传输的对象的类型中有其他的类型,而这些类型并没有被申明的话,那么在执行WriteObject的时候会报错,因为编译器认为类型不明确或者不认识。

这个时候需要为序列化的类型进行类型申明。

[csharp] view plaincopyprint?
  1. [DataContract] 
  2.     [KnownType(typeof(Bitmap))] 
  3.     public class ImageData : ContractData
  4.     { 
  5.         [DataMember] 
  6.         public string imageName {get;set; }
  7.         [DataMember] 
  8.         public Image image { get; set; }
  9.         [DataMember] 
  10.         public int index {get;set; }
  11.     } 
[csharp] view plaincopyprint?
  1. [DataContract] 
  2.     [KnownType(typeof(Bitmap))] 
  3.     public class ImageData : ContractData 
  4.     { 
  5.         [DataMember] 
  6.         public string imageName {get; set; } 
  7.         [DataMember] 
  8.         public Image image { get; set; } 
  9.         [DataMember] 
  10.         public int index {get; set; } 
  11.     } 
[csharp] view plaincopyprint?
  1. 例如上面这个类型,Image类型虽然是类库中的类型,但是WriteObject仍然会报错,因为这个类型没有被申明。
[csharp] view plaincopyprint?
  1. 例如上面这个类型,Image类型虽然是类库中的类型,但是WriteObject仍然会报错,因为这个类型没有被申明。 
[csharp] view plaincopyprint?
  1. 需要使用DataContract、KnownType、DataMember来标明。 
[csharp] view plaincopyprint?
  1. 需要使用DataContract、KnownType、DataMember来标明。 
[csharp] view plaincopyprint?
  1. DataMember是必须要的,否则WriteObject根本不会序列化,传过去也是个空的。 
0 0
原创粉丝点击