Как конвертировать 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-данных. Если это то, что вы хотите, я могу заполнить пробелы выше.

С XPath вы можете выбрать список узлов, выбрав путь.

Другие вопросы по тегам