Wcf序列化的循环引用问题1

1.Wcf数据契约序列化,使用的类DataContractSerializer

默认如果类不指定[DataContract],则序列化类的所有字段,并且在出现循环引用的时候回抛出异常,服务终止

msdn文档说明:https://msdn.microsoft.com/library/system.runtime.serialization.datacontractserializer.aspx

/*
    * Wcf 数据契约序列化使用“DataContractSerializer”,底层是xml序列化
    * 1.如果类上不指定 [DataContract],默认序列化,对象的所有属性(包括null的)
    * 2.如果类上指定 [DataContract],序列化只检测 属性有 [DataMember] 的
    * 3.没有显示指定忽略某字段的方式
    */
//创建班级
Grade gradeOne = new Grade() { GradeID = 1, GradeName = "班级1" };
gradeOne.Students = new List<Student>();
//添加班级下的学生
Student stu = new Student();
stu.ID = 1;
stu.Name = "张三";
//设置学生所属的班级 
/*
    * 1.注意:如果当前子类的父类对象引用了当前父类对象抛出循环引用异常
    * 解决方法1: 对于需要序列化的类显示指定契约标识 [DataContract] 
    * 并且忽略 子类的父类对象,
    * 也就是 对 Grade 不指定 [DataMember]
    */
/*
    * 解决方法2:
    * 在类的方法标记使用  [DataContract(IsReference = true)]
    * IsReference:如果使用标准 XML 保留对象引用数据,则为 true;否则为 false
    */
stu.Grade = gradeOne;
gradeOne.Students.Add(stu);
DataContractSerializer serializer = new DataContractSerializer(gradeOne.GetType());
string result = null;
using (MemoryStream s = new MemoryStream())
{
    serializer.WriteObject(s, gradeOne);
    s.Seek(0, SeekOrigin.Begin);
    using (StreamReader r = new StreamReader(s))
    {
        result = r.ReadToEnd();
    }
}
Console.WriteLine(result);

默认设置(出现异常)

public class Student
{
    public int ID { get; set; }
    public string Name { get; set; }
    public Grade Grade { get; set; }
}
public class Grade
{
    public int GradeID { get; set; }
    public string GradeName { get; set; }
    public List<Student> Students { get; set; }
}

解决方法1,忽略导致循环引用的属性:

这样的缺点:就是在客户端获取到学生对象,不能直接也得到学生所属的班级对象

[DataContract]
public class Student
{
    [DataMember]
    public int ID { get; set; }
    [DataMember]
    public string Name { get; set; }

    [DataMember]
    public Grade Grade { get; set; }
}
[DataContract]
public class Grade
{
    [DataMember]
    public int GradeID { get; set; }
    [DataMember]
    public string GradeName { get; set; }
    [DataMember]
    public List<Student> Students { get; set; }
}

序列化结果:

<?xml version="1.0" encoding="utf-8"?>
<Grade xmlns="http://schemas.datacontract.org/2004/07/Test" xmlns:i="http://www.w3.org/2001/XMLSchema-instance">
  <GradeID>1</GradeID>
  <GradeName>班级1</GradeName>
  <Students>
    <Student>
      <Grade i:nil="true"/>
      <ID>1</ID>
      <Name>张三</Name>
    </Student>
  </Students>
</Grade>

解决方法2(推荐),在DataContract上使用IsReference参数,并且设置为true,

互操作引用说明:https://msdn.microsoft.com/zh-cn/library/cc656708(v=vs.110).aspx

IsReferences:如果使用标准 XML 保留对象引用数据,则为 true;否则为 false。

[DataContract(IsReference = true)]
public class Student
{
    [DataMember]
    public int ID { get; set; }
    [DataMember]
    public string Name { get; set; }
    [DataMember]
    public Grade Grade { get; set; }
}

[DataContract(IsReference = true)]
public class Grade
{
    [DataMember]
    public int GradeID { get; set; }
    [DataMember]
    public string GradeName { get; set; }
    [DataMember]
    public List<Student> Students { get; set; }
}

或者在初始化对象的时候指定

DataContractSerializer serializer = new DataContractSerializer(typeof(Parent),
          "Parent",
          string.Empty,
          null,
          int.MaxValue,
          false,
          true,
          null,
          null);

序列化结果:

<?xml version="1.0" encoding="utf-8"?>
<Grade xmlns="http://schemas.datacontract.org/2004/07/Test" xmlns:i="http://www.w3.org/2001/XMLSchema-instance" xmlns:z="http://schemas.microsoft.com/2003/10/Serialization/" 
z:Id
="i1"> <GradeID>1</GradeID> <GradeName>班级1</GradeName> <Students> <Student z:Id="i2"> <Grade z:Ref="i1"/> <ID>1</ID> <Name>张三</Name> </Student> </Students> </Grade>
posted @ 2016-05-11 16:07  天马3798  阅读(520)  评论(0编辑  收藏  举报