如何反序列化xml字符串

时间:2012-05-29 19:22:06

标签: c# .net xml serialization

我试图反序列化一个xml字符串,而我没有反序列化这些对象。

我的xml字符串看起来像

<Cars>
    <Car>
        <Id>123445</Id>
        <BMW>
          <Id>78945</Id>
        </BMW>
    </Car>
</Cars>
[Serializable()]
[XmlRoot("Cars")]
public class Cars
{
    [XmlArrayItem("Car",typeof(Car))]
    public Car[] Car { get; set; }
}

[Serializable()]
public class Car
{
    [XmlElement("Id")]
    public long Id { get; set; }

    [XmlArrayItem("BMW",typeof(BMW))]
    public BMW[] BMW { get; set; }
}

[Serializable()]
public class BMW
{
    [XmlElement("Id")]
    public long Id { get; set; }
}

这是我正在尝试的代码:

string str = "xml string like above";
XmlSerializer ser = new XmlSerializer(typeof(Cars));
var wrapper = (Cars) ser.Deserialize(new StringReader(str));

我确实有像宝马里面的几个子阵列对象。不,它不是序列化汽车。 有人可以在这里指出我的错误吗?

4 个答案:

答案 0 :(得分:3)

尝试使用XmlElement代替XmlArrayItem,因为这些项目没有封装标记:

[Serializable()]
[XmlRoot("Cars")]
public class Cars
{
    [XmlElement("Car",typeof(Car))]
    public Car[] Car { get; set; }
}

[Serializable()]
public class Car
{
    [XmlElement("Id")]
    public long Id { get; set; }

    [XmlElement("BMW",typeof(BMW))]
    public BMW[] BMW { get; set; }
}

答案 1 :(得分:2)

尝试将预期对象序列化为xml并进行比较。 (序列化方法比反序列化方法更容易。)这可能会突出显示错误放置或错误命名的属性。

答案 2 :(得分:0)

我有几个建议和一个很好的工作示例。首先,我建议使用不同的类结构来利用类层次结构。

以下是您的需求:

  • CarList - 用于管理您的汽车列表的包装类
  • Car - 提供所有基本Car属性和方法的类
  • 宝马 - 特定类型的汽车级别。它继承了父Car类并提供了其他属性和方法。

当你想要添加更多Car类型时,你会创建另一个类并让它继承Car类。

代码......

这是你的CarList课程。请注意,Car对象列表上有多个XmlElement声明。这样序列化/反序列化就知道如何处理不同的Car类型。当您添加更多类型时,您必须在此处添加一行来描述类型。

另外,请注意处理序列化/反序列化的静态Save和Load方法。

[XmlRoot("CarList")]
public class CarList
{
  [XmlAttribute("Name")]
  public string Name  { get; set; }

  [XmlElement("Cars")]
  [XmlElement("BWM", typeof(BMW))]
  [XmlElement("Acura", typeof(Acura))]
  //[XmlArrayItem("Honda", typeof(Honda))]
  public List<Car> Cars  { get; set; }

  public static CarList Load(string xmlFile)
  {
    CarList carList = new CarList();
    XmlSerializer s = new XmlSerializer(typeof(CarList));
    TextReader r = new StreamReader(xmlFile);
    carList = (CarList)s.Deserialize(r);
    r.Close();
    return carList;
  }

  public static void Save(CarList carList, string fullFilePath)
  {
    XmlSerializer s = new XmlSerializer(typeof(CarList));
    TextWriter w = new StreamWriter(fullFilePath);

    // use empty namespace to remove namespace declaration
    XmlSerializerNamespaces ns = new XmlSerializerNamespaces();
    ns.Add("", "");

    s.Serialize(w, carList, ns);
    w.Close();
  }
}

接下来,这是你的Car class ......

public class Car
{
  // put properties and methods common to all car types here
  [XmlAttribute("Id")]
  public long Id { get; set; }
}

最后,您的特定车型......

public class BMW : Car
{
  // put properties and methods specific to this type here
  [XmlAttribute("NavVendor")]
  public string navigationSystemVendor { get; set; }
}

public class Acura : Car
{
  // put properties and methods specific to this type here
  [XmlAttribute("SunroofTint")]
  public bool sunroofTint { get; set; }
}

像这样加载您的CarList:

CarList carList = CarList.Load(@"C:\cars.xml");

进行一些更改,然后像这样保存您的CarList:

CarList.Save(carList, @"C:\cars.xml");

这就是XML的样子:

<?xml version="1.0" encoding="utf-8"?>
<CarList Name="My Car List">
  <BWM Id="1234" NavVendor="Alpine" />
  <Acura Id="2345" SunroofTint="true" />
</CarList>

希望有助于您开始朝着正确的方向前进!

答案 3 :(得分:0)

我想我发现了问题。我没有包括我在xml中获取的所有属性。如果您错过任何属性,反序列化将不起作用。 我还添加了这样一行:

[XmlArray(Form = XmlSchemaForm.Unqualified),
     XmlArrayItem("BMW", typeof(BMW), Form = XmlSchemaForm.Unqualified, IsNullable = false)]
    public BMW[] BMWs { get; set; }

如果你的xml有这样的东西:

<Cars>
  <Car>
    <Id>123445</Id>
    <BMWs>
       <BMW>
          <Id>78945</Id>
       <BMW>
    </BMWs>
  </Car>
</Cars>

然后它奏效了。我没有任何XSD或模式来生成我认为理想的对象。