XML解析 - 读取简单的XML文件并检索值

时间:2011-04-09 10:02:57

标签: c# .net xml parsing

我已经编写了一个用于学习目的的任务调度程序。目前我正在将计划任务保存为纯文本,然后使用Regex进行解析。这看起来很混乱(代码明智)并且不是非常连贯。

我想从XML文件加载计划任务,我已经搜索了很多以找到一些解决方案,但我无法按照我想要的方式工作。

我写了一个像这样结构化的XML文件来存储我的数据:

<Tasks>
    <Task>
        <Name>Shutdown</Name>
        <Location>C:/WINDOWS/system32/shutdown.exe</Location>
        <Arguments>-s -f -t 30</Arguments>
        <RunWhen>
            <Time>8:00:00 a.m.</Time>
            <Date>18/03/2011</Date>
            <Days>
                <Monday>false</Monday>
                <Tuesday>false</Tuesday>
                <Wednesday>false</Wednesday>
                <Thursday>false</Thursday>
                <Friday>false</Friday>
                <Saturday>false</Saturday>
                <Sunday>false</Sunday>
                <Everyday>true</Everyday>
                <RunOnce>false</RunOnce>
            </Days>
        </RunWhen>
        <Enabled>true</Enabled>
    </Task>
</Tasks>

我想解析数据的方式如下:

  
      
  1. 打开Tasks.xml
  2.   
  3. 加载第一个任务标记。
  4.   
  5. 在该任务中检索Name,Location和Arguments标记的值。
  6.   
  7. 然后打开RunWhen标记并检索时间和日期标记的值。
  8.   
  9. 之后打开Days标记并检索其中每个标记的值。
  10.   
  11. 检索已启用的值。
  12.   
  13. 加载下一个任务并重复步骤3 - &gt; 7直到任务中的所有任务标签都已被解析。
  14.   

我非常肯定你可以这样做,我无法解决这个问题,因为有很多不同的方法可以用XML做事我有点不知所措。但到目前为止,我最有可能的是使用XPathDocument和XPathNodeIterator吗?

如果有人可以向我展示一个例子或向我解释如何做到这一点我将非常高兴。

5 个答案:

答案 0 :(得分:44)

解析xml的简便方法是使用LINQ to XML

例如,您有以下xml文件

<library>
    <track id="1" genre="Rap" time="3:24">
        <name>Who We Be RMX (feat. 2Pac)</name>
        <artist>DMX</artist>
        <album>The Dogz Mixtape: Who's Next?!</album>
    </track>
    <track id="2" genre="Rap" time="5:06">
        <name>Angel (ft. Regina Bell)</name>
        <artist>DMX</artist>
        <album>...And Then There Was X</album>
    </track>
    <track id="3" genre="Break Beat" time="6:16">
        <name>Dreaming Your Dreams</name>
        <artist>Hybrid</artist>
        <album>Wide Angle</album>
    </track>
    <track id="4" genre="Break Beat" time="9:38">
        <name>Finished Symphony</name>
        <artist>Hybrid</artist>
        <album>Wide Angle</album>
    </track>
<library>

要阅读此文件,您可以使用以下代码:

public void Read(string  fileName)
{
    XDocument doc = XDocument.Load(fileName);

    foreach (XElement el in doc.Root.Elements())
    {
        Console.WriteLine("{0} {1}", el.Name, el.Attribute("id").Value);
        Console.WriteLine("  Attributes:");
        foreach (XAttribute attr in el.Attributes())
            Console.WriteLine("    {0}", attr);
        Console.WriteLine("  Elements:");

        foreach (XElement element in el.Elements())
            Console.WriteLine("    {0}: {1}", element.Name, element.Value);
    }
}

答案 1 :(得分:11)

我通常会使用XmlDocument。界面非常简单:

var doc = new XmlDocument();
doc.LoadXml(xmlString);

您可以访问类似于字典的节点:

var tasks = doc["Tasks"];

并循环遍历节点的所有子节点。

答案 2 :(得分:2)

您熟悉DataSet班吗?

DataSet也可以load个XML文档,您可能会发现迭代更容易。

http://msdn.microsoft.com/en-us/library/system.data.dataset.readxml.aspx

DataSet dt = new DataSet();
dt.ReadXml(@"c:\test.xml");

答案 3 :(得分:2)

尝试XmlSerialization

试试这个

[Serializable]
public class Task
{
    public string Name{get; set;}
    public string Location {get; set;}
    public string Arguments {get; set;}
    public DateTime RunWhen {get; set;}
}

public void WriteXMl(Task task)
{
    XmlSerializer serializer;
    serializer = new XmlSerializer(typeof(Task));

    MemoryStream stream = new MemoryStream();

    StreamWriter writer = new StreamWriter(stream, Encoding.Unicode);
    serializer.Serialize(writer, task);

    int count = (int)stream.Length;

     byte[] arr = new byte[count];
     stream.Seek(0, SeekOrigin.Begin);

     stream.Read(arr, 0, count);

     using (BinaryWriter binWriter=new BinaryWriter(File.Open(@"C:\Temp\Task.xml", FileMode.Create)))
     {
         binWriter.Write(arr);
     }
 }

 public Task GetTask()
 {
     StreamReader stream = new StreamReader(@"C:\Temp\Task.xml", Encoding.Unicode);
     return (Task)serializer.Deserialize(stream);
 }

答案 4 :(得分:0)

class Program
{

    static void Main(string[] args)
    {

        //Load XML from local
        string sourceFileName="";
        string element=string.Empty;
        var FolderPath=@"D:\Test\RenameFileWithXmlAttribute";

            string[] files = Directory.GetFiles(FolderPath, "*.xml");
            foreach (string xmlfile in files)
            {
                try
                {
                    sourceFileName = xmlfile;
                    XElement xele = XElement.Load(sourceFileName);
                    string convertToString = xele.ToString();
                    XElement parseXML = XElement.Parse(convertToString);
                    element = parseXML.Descendants("Meta").Where(x => (string)x.Attribute("name") == "XMLTAG").Last().Value;
                    DirectoryInfo CurrentDate = Directory.CreateDirectory(DateTime.Now.ToString("yyyy-MM-dd"));
                    string saveWithThisName= Path.Combine(CurrentDate.FullName, element);
                    File.Copy(sourceFileName, saveWithThisName,true);                      
                }
                catch(Exception ex)
                {

                }      
            }       
    }
}