我试图反序列化一个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));
我确实有像宝马里面的几个子阵列对象。不,它不是序列化汽车。 有人可以在这里指出我的错误吗?
答案 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)
我有几个建议和一个很好的工作示例。首先,我建议使用不同的类结构来利用类层次结构。
以下是您的需求:
当你想要添加更多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或模式来生成我认为理想的对象。