Webservice SOAP传输序列化总结 以及webservice之序列化以及反序列化实例

时间:2021-11-01 12:39:21

一、所有Webservice中传递的对象都必须能够序列化,这个是作为在网络之间传输的必要条件。XML WebService和SOAP标准支持的数据类型如下:

1.基本数据类型.

标准类型,如:int float bool DateTime string等基本数据类型

2.枚举.

支持枚举Enum定义的类型

3.自定义对象.

可以传递任意基于自定义类或结构创建的对象。 但要注意一点: 它只能传输数据成员(变量和属性).

如果定义了方法,则方法不能进行序列化传输,序列化后只剩下数据成员.

4.DataSet对象

支持DataSet,切记:不支持DataTable和DataRow,DataSet已经是XML Webservice能够支持的最小的可序列化对象.

5.XmlNode对象

基于XmlNode的对象可以表示XML文档的一部分.

6.数组和集合

可以使用任何被支持的类型的数组和简单集合,包括: DataSet对象/XmlNode对象和自定义对象.

二、webservice传输过程中常遇到的问题

  (1)在调用WebService并给一个方法传递了一个DataRow参数时,运行时会抛出异常: "没法将参数序列化!",如果把DataRow加入到DataSet中,并将DataSet作为参数          传递再运行就OK了。 这是因为:XML WebService只能对数据集DataSet对象类型进行XML序列化,不能对DataRow对象类型进行XML序列化造成的错误.因此了解一下               XML WebService支持序列化的基本数据类型是比较重要的.
  (2)除了上述的基本类型以外,Webservice服务不能完成直接序列化传输,但是大部分数据结构借助序列化,仍然可以在Webservice上传输。

     例如:

     在将一个自定义类序列化到文件时,出现如下错误提示:

     System.Reflection.TargetInvocationException: 调用的目标发生了异常。 --->

     System.InvalidOperationException: 生成 XML 文档时出错。 ---> System.InvalidOperationException:

     不应是类型 Alink.T1System.Windows.PrintSetting。使用 XmlInclude 或 SoapInclude 属性静态指定非已知的类型。

      经查找资料,发现当被序例化的类中包含自定义的复杂类时,需要使用XmlIncludeAttribut属性标识自定义类

     现在修改如下:

     using System.Xml.Serialization;

     [XmlInclude(typeof(自定义类))]

     [Serializable]

     public class MyDIYClass

     自定义的类都添加标记,提供的webservice方法也加标记

     [WebMethod]

     [XmlInclude(typeof(自定义类))]

     public object WebServiceMethod(Object requestObj)

     要把该类中内含的自定义类都用XmlInclude一次,就可以解决问题了;

    这样就可以传输了,List<>,ArrayList对象都可以传输,但是返回类型为ArrayList的[WebMethod]方法,在客户端调用后,得到的是object数组,而且object中的             property全部都变为field。另外,接口也不能序列化例如List<Interface>也是不能序列化的,遇到这种情况可以把接口修改为抽象类,同时在抽象类上添加[XmlInclude]属      性,最后需要注意一点,自己写的类要想序列化必须有默认构造函数(不带参数的构造函数)。

    三、xml序列化答疑:

  (1)需序列化的字段必须是公共的(public)

  (2)需要序列化的类都必须有一个无参的构造函数

  (3)枚举变量可序列化为字符串,无需用[XmlInclude]

  (4)导出非基本类型对象,都必须用[XmlInclude]事先声明。该规则递归作用到子元素

  如导出ArrayList对象,若其成员是自定义的,需预包含处理:

  using System.Xml.Serialization;

  [XmlInclude(typeof(自定义类))]

  (5)Attribute中的IsNullable参数若等于false,表示若元素为null则不显示该元素。

  也就是说:针对值类型(如结构体)该功能是实效的

  若数组包含了100个空间,填充了10个类对象,则序列化后只显示10个节点

  若数组包含了100个空间,填充了10个结构体对象,则序列化后会显示100个节点

  (6)真正无法XML序列化的情况,某些类就是无法XML序列化的(即使使用了[XmlInclude])

  IDictionary(如HashTable)

  System.Drawing.Color

  System.Drawing.Font

  SecurityAttribute声明

  父类对象赋予子类对象值的情况

  对象间循环引用

  (7)对于无法XML序列化的对象,可考虑使用自定义xml序列化(实现IXmlSerializable接口)实现IDictionary的类,可考虑

      (1)用其它集合类替代;(2)用类封装之,并提供Add和this函数

   (8)某些类型需要先经过转换,然后才能序列化为 XML。如XML序列化System.Drawing.Color,可先用ToArgb()将其转换为整数,过于复杂的对象用xml序列化不便的                    话,可考虑用二进制序列化。

  四、webservice序列化和反序列化实例

    //序列化为xml的方法    

    public static string ToXml(this object obj)
        {
            XmlSerializer serializer = new XmlSerializer(obj.GetType());
            StringWriter sw = new StringWriter();
                serializer.Serialize(sw, obj);
            return sw.ToString();
        }

    //反序列化为XML的方法

    public static T ToObj<T>(string xml) //where T : Object
{
XmlSerializer serializer = new XmlSerializer(typeof(T));
StringReader sw = new StringReader(xml);
object obj = serializer.Deserialize(sw);
if (obj is T)
{
return (T)obj;
}
else
{
return default(T);
}
}

    //客户端调用webservice中的InsertData(string entity)方法,传递的是的类实体product_entity,序列化代码如下:    

      pm.WebProductService ps = new pm.WebProductService(); 
      Entity.VAERP.Product product_entity = new Entity.VAERP.Product();
      string entity = product_entity.ToXml();
string result = ps.InsertData(entity);

    //服务端webservice中反序列化,代码示例如下:    

      //entity是客户端传过来的序列化的参数
      Entity.VAERP.Product product_entity = new Entity.VAERP.Product();
      product_entity = ToObj<Entity.VAERP.Product>(entity);