Как конвертировать xml в список объектов
Вот код, моя цель - перебрать все узлы, а не только один, поэтому в моем случае он получает первый узел устройства и всех его дочерних элементов одновременно, так как я могу получить все узлы устройства и PortA PortB. тогда я смогу установить значения моего класса.
private void loadXmlFile(string path)
{
try
{
XElement deviceElement = XElement.Load(path);
Device device = new Device();
var allElements = deviceElement.DescendantNodes();
foreach (var elm in deviceElement.Elements())
{
if (elm.Name == "Device")
{
foreach (XAttribute att in elm.Attributes())
{
if (att.Name == "Type")
{
device.TBType1 = att.Value.ToString();
}
if (att.Name == "Name")
{
device.Name = att.Value.ToString();
}
if (att.Name == "ParentConnectedToPort")
{
device.ParentConnectedTo = att.Value.ToString();
}
if (att.Name == "DeviceConnectedToPort")
{
device.DeviceConnectedTo = att.Value.ToString();
}
string connectedTo = (string)elm.Parent.Attribute("Connected_BY");
if (att.Name == "properties" && connectedTo == "Directly")
{
string str = att.Value;
string[] parts = str.Split(':', ',');
}
}
}
}
}
catch (Exception ex)
{
throw ex;
}
}
Вот мой XML-файл выглядит так, как будто нужно получить каждый элемент устройства и его собственные значения и установить значения моего класса.
<CMS>
<Device Name="CM_HOST" Type="TR">
<PortA Connected_BY="Directly">
<Device TB="AR" ParentConnectedToPort="A" Name="Akitio" DeviceConnectedToPort="A" properties="{'Type': 'AR' ,'Name': 'Akitio','Cable': '20G Passive'}">
<PortB Connected_BY="Directly">
<Device TB="AR" ParentConnectedToPort="A" Name="doc1" DeviceConnectedToPort="B" properties="{'Type': 'AR' ,'Name': 'doc1','Cable': '20G Passive'}">
<PortB Connected_BY="Directly">
<Device TB="AR" ParentConnectedToPort="A" Name="doc2" DeviceConnectedToPort="B" properties="{'Type': 'AR' ,'Name': 'doc2','Cable': '20G Passive'}" />
</PortB>
<PortA Connected_BY="None" />
<PortE Connected_BY="None" />
</Device>
</PortB>
<PortA Connected_BY="None" />
<PortE Connected_BY="None" />
</Device>
</PortA>
<PortB Connected_BY="None" />
<PortE Connected_BY="None" />
</Device>
<Sx properties="{'FTDI_port': '0' ,'SX_power_button_pin': '7','SX_SLP_S3_pin': '6','SX_SLP_S4_pin': '11','SX_SLP_S5_pin': '10','SX_TBT_wake_N_pin': '8','SX_PCIE_wake_pin': '9','G3_Power_Mode': 'PowerSplitter'}" />
</CMS>
3 ответа
Мне нравится использовать рекурсивный алгоритм:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Xml;
using System.Xml.Linq;
using System.Text.RegularExpressions;
namespace ConsoleApplication13
{
class Program
{
const string FILENAME = @"c:\temp\test.xml";
static void Main(string[] args)
{
XDocument doc = XDocument.Load(FILENAME);
XElement cms = doc.Descendants("CMS").FirstOrDefault();
XElement device = cms.Element("Device");
Device.RecursiveParseXml(device, Device.root);
}
}
public class Device
{
public static Device root = new Device();
public List<Port> ports { get; set; }
public string name { get; set; }
public string type { get; set; }
public string tb { get; set; }
public string deviceConnectedToPort { get; set; }
public Dictionary<string, string> properties { get; set; }
public static void RecursiveParseXml(XElement parent, Device device)
{
device.name = (string)parent.Attribute("Name");
device.type = (string)parent.Attribute("Type");
device.tb = (string)parent.Attribute("TB");
device.deviceConnectedToPort = (string)parent.Attribute("DeviceConnectedToPort");
string strProperties = (string)parent.Attribute("properties");
if (strProperties != null)
{
string[] propertyArray = strProperties.Split(new char[] { ',', '{', '}' }, StringSplitOptions.RemoveEmptyEntries).ToArray();
string pattern = @"'(?'name'[^']+)':\s+'(?'value'[^']+)";
device.properties = new Dictionary<string, string>();
foreach (string property in propertyArray)
{
Match match = Regex.Match(property, pattern);
device.properties.Add(match.Groups["name"].Value, match.Groups["value"].Value);
}
}
foreach (XElement element in parent.Elements())
{
Port newPort = new Port();
if (device.ports == null)
{
device.ports = new List<Port>();
}
device.ports.Add(newPort);
newPort.connectBy = (string)element.Attribute("Connected_BY");
XElement newDevice = element.Element("Device");
if (newDevice != null)
{
newPort.device = new Device();
RecursiveParseXml(element.Element("Device"), newPort.device);
}
}
}
}
public class Port
{
public string name { get; set; }
public string connectBy { get; set; }
public Device device { get; set; }
}
}
Сериализация XML может сделать то, что вы хотите. Не совсем уверен, каким должен быть ваш конечный результат. Вы можете использовать классы ниже, чтобы просто прочитать данные XML и затем создать нужные вам объекты из результатов.
Создайте классы.
public class CMS
{
[XmlElement("Device")]
List<Device> Devices {get;set;}
//other properties here...
}
public class Device
{
public Port PortA { get;set;}
public Port PortB { get;set;}
public Port PortC { get;set;}
public Port PortD { get;set;}
public Port PortE { get;set;}
//other properties here...TB, ParentConnectedToPort etc
}
public class Port
{
public Device Device { get; set; }
//other properties here...Connected_BY etc
}
Затем вы можете использовать эту функцию для десериализации из строки:
public static T DeserializeXml<T>(string str)
{
var serializer = new XmlSerializer(typeof(T));
object result;
using (TextReader reader = new StringReader(str))
{
result = serializer.Deserialize(reader);
}
return (T) result;
}
Используется таким образом:
var s = "--your xml string--";
var obj = Deserialize<CMS>(s);
Затем obj будет полностью заполненным объектом из ваших XML-данных. Если это то, что вы хотите, я могу заполнить пробелы выше.