一、数据类型的等效性

例如下面定义的两个类成员名称、定义顺序都不一样,但是在DataContract、DataMember的Name属性作用下,两个类的实例对象序列化后的xml是一样的,因此Order和OrderV2对于DataContractSerializer序列化是等效的。

   [DataContract]
public class Order
{
[DataMember]
public int price; [DataMember(Name = "datev2")]
public DateTime date;
} [DataContract(Name = "Order")]
public class OrderV2
{
[DataMember]
public DateTime datev2; [DataMember(Name = "price")]
public int PriceV2;
}

二、数据添加导致的不一致

假设客户端定义的是OrderLess,服务端定义的是OrderMore。客户端会基于OrderLess进行序列化后发送到服务端。服务端会基于OrderMore反序列化来自客户端的OrderLess序列化后的消息,能序列化成功,但是会发现缺少Name的值。

 [DataContract(Name="Order")]
public class OrderLess
{
[DataMember]
public int Price; [DataMember]
public DateTime Date;
} [DataContract(Name = "Order")]
public class OrderMore
{
[DataMember]
public DateTime Date; [DataMember]
public int Price; [DataMember]
public string Name;
}

此时如果我们想对这个缺失的值设置一个默认值,可以通过注册序列化回调方法实现。

OnDeserialized  :  在对象反序列化后立即调用。

OnDeserializing :反序列化对象之前调用。

OnSerialized     :  在序列化后调用此。

OnSerializing    :在序列化前调用。

对OrderMore修改后如下:

[DataContract(Name = "Order")]
public class OrderMore
{
[DataMember]
public DateTime Date; [DataMember]
public int Price; [DataMember]
public string Name; [OnDeserializing]
void OnSerializing(StreamingContext context)
{
this.Name="未指定";
}
}

Main函数:

 static void Main(string[] args)
{
OrderLess less=new OrderLess();
less.Date=DateTime.Now;
less.Price=8;
Serialize<OrderLess>(less,@"D:\1.txt");
OrderMore more=null;
DeSerialize<OrderMore>(@"D:\1.txt",out more);
Console.WriteLine("{0} {1} {2}",more.Date,more.Price,more.Name);
}
public static void Serialize<T>(T t, string path)
{
DataContractSerializer ser = new DataContractSerializer(typeof(T));
using (XmlTextWriter writer = new XmlTextWriter(path, Encoding.UTF8))
{
writer.Formatting = Formatting.Indented;
ser.WriteObject(writer, t);
}
}
public static void DeSerialize<T>(string path, out T t)
{
DataContractSerializer ser = new DataContractSerializer(typeof(T));
using (XmlReader reader = new XmlTextReader(path))
{
t = (T)ser.ReadObject(reader);
}
}

输出:

三、数据删除导致的不一致

与二中的交换一下,客户端定义的是OrderMore,服务端定义的是OrderLess。客户端基于OrderMore类型的序列化器序列化OrderMore对象,服务端按基于OrderLess的序列化器反序列化来自客户端的OrderMore的消息。服务端也能序列化成功,但是当服务端将这个对象返回给客户端时,客户端那边会发现少了Name值。为了避免这种情况发生,OrderLess可以实现IExtensibleDataObject接口。作用是将反序列化时未知的数据存放到ExtensionData,再次对这个对象序列化时就可以还原这些未知对象。

对OrderLess进行修改:

   [DataContract(Name="Order")]
public class OrderLess:IExtensibleDataObject
{
[DataMember]
public int Price; [DataMember]
public DateTime Date; public ExtensionDataObject ExtensionData
{
get;set;
}
}

Main函数模拟客户端和服务端的交互过程,先序列化OrderMore,再反序列化为OrderLess,此时将未知的Name属性放在OrderLess的ExtensionData。再次序列化这个OrderLess对象。发现1.txt和2.txt的序列化内容是一样的。

 static void Main(string[] args)
{
OrderMore more = new OrderMore();
more.Date = DateTime.Now;
more.Price = 8;
more.Name = "OrderMore";
Serialize<OrderMore>(more, @"D:\1.txt"); OrderLess less = null;
DeSerialize<OrderLess>(@"D:\1.txt", out less);
Serialize<OrderLess>(less, @"D:\2.txt");
}

四、数据契约代理

在上述描述都是数据成员添加或删除这种小差异。当两个对象差异很大时(如下面的A和B),却想让两个对象序列化时等效,需要数据契约代理来做一些工作。

[DataContract]
public class A
{
[DataMember]
public string FullName; [DataMember]
public int age;
} [DataContract]
public class B
{
[DataMember]
public string FirstName; [DataMember]
public string LastName; [DataMember]
public int age;
}

MatchA2B实现IDataContractSurrogate接口,在DataContractSerializer序列化过程中能够将一个A对象序列化为B对象,将B对象反序列化为A对象。需要实现GetDataContractType方法,获取需要序列化的实际类型。GetObjectToSerialize序列化前的转换,GetDeserializedObject序列化后的转换。

public class MatchA2B : IDataContractSurrogate
{
public Type GetDataContractType(Type type)
{
return type == typeof(A) ? typeof(B) : type;
}
public object GetObjectToSerialize(object obj, Type targetType)
{
A a=obj as A;
if (a != null)
{
B b = new B();
b.age = a.age;
b.FirstName = a.FullName.Split(" ".ToCharArray())[0];
b.LastName = a.FullName.Split(" ".ToCharArray())[1];
return b;
}
return obj;
} public object GetDeserializedObject(object obj, Type targetType)
{
B b = obj as B;
if (b != null)
{
A a = new A();
a.FullName = b.FirstName + " " + b.LastName;
a.age = b.age;
return a;
}
return obj;
}
public object GetCustomDataToExport(Type clrType, Type dataContractType){return null;}
public object GetCustomDataToExport(System.Reflection.MemberInfo memberInfo, Type dataContractType){return null;}
public void GetKnownCustomDataTypes(System.Collections.ObjectModel.Collection<Type> customDataTypes){ }
public Type GetReferencedTypeOnImport(string typeName, string typeNamespace, object customData){return null;}
public System.CodeDom.CodeTypeDeclaration ProcessImportedType(System.CodeDom.CodeTypeDeclaration typeDeclaration, System.CodeDom.CodeCompileUnit compileUnit)
{ return null; }
}

Main函数:

class Program
{
static void Main(string[] args)
{
A a = new A();
a.FullName = "Leonardo DiCaprio";
a.age = 41;
Serialize<A>(a, @"D:\1.txt"); A b = null;
DeSerialize<A>(@"D:\1.txt", out b);
Console.WriteLine("{0} {1} ", b.FullName, b.age);
}
public static void Serialize<T>(T t, string path)
{
DataContractSerializer ser = new DataContractSerializer(typeof(T),null,int.MaxValue,false,false,new MatchA2B());
using (XmlTextWriter writer = new XmlTextWriter(path, Encoding.UTF8))
{
writer.Formatting = Formatting.Indented;
ser.WriteObject(writer, t);
}
}
public static void DeSerialize<T>(string path, out T t)
{
DataContractSerializer ser = new DataContractSerializer(typeof(T), null, int.MaxValue, false, false, new MatchA2B());
using (XmlReader reader = new XmlTextReader(path))
{
t = (T)ser.ReadObject(reader);
}
}
}

1.txt中的内容为,成功的将A对象序列化为B:

<B xmlns:i="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://schemas.datacontract.org/2004/07/ConsoleApplication3">
<FirstName>Leonardo</FirstName>
<LastName>DiCaprio</LastName>
<age>41</age>
</B>

DataContractSerializer数据不一致下序列化的更多相关文章

  1. c#中@标志的作用 C#通过序列化实现深表复制 细说并发编程-TPL 大数据量下DataTable To List效率对比 【转载】C#工具类:实现文件操作File的工具类 异步多线程 Async .net 多线程 Thread ThreadPool Task .Net 反射学习

    c#中@标志的作用   参考微软官方文档-特殊字符@,地址 https://docs.microsoft.com/zh-cn/dotnet/csharp/language-reference/toke ...

  2. Redis面试题记录--缓存双写情况下导致数据不一致问题

    转载自:https://blog.csdn.net/lzhcoder/article/details/79469123 https://blog.csdn.net/u013374645/article ...

  3. 重温WCF之数据契约和序列化(四)

    一.数据契约 1.使用数据协定可以灵活控制哪些成员应该被客户端识别. [DataContract] public class Employee { [DataMember] public string ...

  4. 跟我一起学WCF(7)——WCF数据契约与序列化详解

    一.引言 在前面博文介绍到,WCF的契约包括操作契约.数据契约.消息契约和错误契约,前面一篇博文已经结束了操作契约的介绍,接下来自然就是介绍数据契约了.所以本文要分享的内容就是数据契约. 二.数据契约 ...

  5. WCF分布式开发步步为赢(7):WCF数据契约与序列化

    本节继续学习WCF分布式开发步步为赢(7):WCF数据契约与序列化.数据契约是WCF应用程序开发中一个重要的概念,毫无疑问实现客户端与服务端数据契约的传递中序列化是非常重要的步骤.那么序列化是什么?为 ...

  6. mysql]一次主从数据不一致的问题解决过程()

    问题 要解决问题就是怎么对比不一致,然后在不影响业务的情况下,修复数据不一致的问题,把从库缺少的数据补上 下面是能想到和找到的几个方案 1 从新从0开始同步,虽然对主库的使用没有影响,但是那么大的数据 ...

  7. MySQL主从复制数据不一致问题【自增主键】

    前言: 今天遇到主从表不一致的情况,很奇怪为什么会出现不一致的情况,因为复制状态一直都是正常的.最后检查出现不一致的数据都是主键,原来是当时初始化数据的时候导致的.现在分析记录下这个问题,避免以后再遇 ...

  8. MySQL多字节字符集造成主从数据不一致问题

    MySQL多字节字符集造成主从数据不一致问题 来自江羽   2013-04-27 16:03:56|  分类: 默认分类|举报|字号 订阅 转载: http://backend.blog.163.co ...

  9. [mysql]一次主从数据不一致的问题解决过程

    之前一篇: 主从更换ip之后重新建立同步 情况时这样的 昨天晚上主动2个机器都迁移了,然后今天才把主动重新连接上,但是从库的偏移量是从今天当前时刻开始的,也就是说虽然现在主动看似正常,其实是少了昨天的 ...

随机推荐

  1. C#通过盘符获取剩余空间

    public static long GetHardDiskSpace(string str_HardDiskName) { ; str_HardDiskName = str_HardDiskName ...

  2. Docker容器的原理与实践(上)

    本文来自网易云社区. 虚拟化 是一种资源管理技术,将计算机的各种资源予以抽象.转换后呈现出来, 打破实体结构间的不可切割的障碍,使用户可以比原本更好的方式来应用这些资源. Hypervisor 一种运 ...

  3. Day 39 管道 、数据共享与地址池

    参考张磊同学的博客 http://www.cnblogs.com/chongdongxiaoyu/p/8658379.html 一.管道 #创建管道的类: Pipe([duplex]):在进程之间创建 ...

  4. LOJ#3088. 「GXOI / GZOI2019」旧词(树剖+线段树)

    题面 传送门 题解 先考虑\(k=1\)的情况,我们可以离线处理,从小到大对于每一个\(i\),令\(1\)到\(i\)的路径上每个节点权值增加\(1\),然后对于所有\(x=i\)的询问查一下\(y ...

  5. 深入理解Java接口和抽象类

    对于面向对象编程来说,抽象是它的一大特征之一.在Java中,可以通过两种形式来体现OOP的抽象:接口和抽象类.这两者有太多相似的地方,又有太多不同的地方.很多人在初学的时候会以为它们可以随意互换使用, ...

  6. leetcode-908-最小差值 I

    题目描述: 给定一个整数数组 A,对于每个整数 A[i],我们可以选择任意 x 满足 -K <= x <= K,并将 x 加到 A[i] 中. 在此过程之后,我们得到一些数组 B. 返回  ...

  7. 前端的CRUD增删改查的小例子

    前端的CRUD增删改查的小例子 1.效果演示 2.相关代码: <!DOCTYPE html> <html lang="en"> <head> & ...

  8. PHP 修改目录下所有与文件夹重名的前缀文件为index.后缀

    <?phpset_time_limit(0); function traverse($path = '.' , $dir_name='') { $current_dir = opendir($p ...

  9. Ubuntukylin-14.04-desktop( 不带分区)安装步骤详解

    不多说,直接上干货! Ubuntukylin-14.04-desktop(带分区)安装步骤详解 Ubuntu14.04安装之后的一些配置 Ubuntukylin-14.04-desktop( 不带分区 ...

  10. 【转】Spark源码分析之-scheduler模块

    原文地址:http://jerryshao.me/architecture/2013/04/21/Spark%E6%BA%90%E7%A0%81%E5%88%86%E6%9E%90%E4%B9%8B- ...