从ListBox更新XML LINQ

时间:2013-06-24 19:27:18

标签: c# linq-to-xml

我在发现如何使用LINQ使用列表框中的值更新XML文件时遇到了一些麻烦。

XML文件如下所示:

<?xml version="1.0" encoding="utf-8"?>
<scripts>
  <script id="3">    
    <name>Test</name>
    <author>Test Author</author>
    <filestoProcess>
      <file>testfile.jpg</file>
      <file>testfile1.jpg</file>
    </filestoProcess>
  </script>
</scripts>

我正在使用:

更新“name”和“author”字段
    XDocument xDoc = XDocument.Load("..\\..\\XMLTest.xml");
    XElement uElem = xDoc.Descendants("script").Where(t => (int)t.Attribute("id") == id).FirstOrDefault();           
    uElem.SetElementValue("name", txtScriptTitle.Text);
    uElem.SetElementValue("author", cmboAuthor.Text);

如何将ListBox值与XML文件中“file”元素的值进行比较,然后相应地添加/删除/更新XML文档中的“文件”值?

更新 可以更容易地从XML中删除“文件”元素,然后添加列表框中保存的值,而不是比较两个列表。

使用此代码我可以删除所有“文件”元素,但是当我尝试从列表框中添加值时,只有最后一个值被写入XML文件。

        // Remove all "file" elements from XML
        uElem.Descendants("file").ToList().ForEach(i => i.Remove());

        // Now add the elements from the listbox
        foreach (string s in lbFilesToProcess.Items)
        {
            uElem.Element("filestoProcess").SetElementValue("file", s);
        }

例如我的列表框包含:file2.jpg,file2.jpg和file3.jpg,但是当我在执行上面的代码后查看XML文件时,我只看到file3.jpg。

知道为什么这不会写出列表框中的所有三个项目?

2 个答案:

答案 0 :(得分:0)

  1. 我会逐步执行您的代码,以确保它循环遍历每个文件。还要仔细检查并确保将“s”设置为正确的文件。

  2. 如果它循环遍历每个文件并成功调用“SetElementValue”,则可能与必须为“foreach”中的每个循环检索“filesToProcess”节点有关。试试这个,而不是你拥有的东西:

    // Now add the elements from the listbox
    var filesToProcessNode = uElem.Element("filestoProcess");
    foreach (string s in lbFilesToProcess.Items)
    {
        filestoProcessNode.SetElementValue("file", s);
    }
    
  3. 我的想法是,因为你使用的是Linq,它可能是一个时间问题,它如何提交对文件的更改,也许它只是在每次循环时覆盖操作,这就是为什么只有最后一个文件使它

    HTH。

答案 1 :(得分:0)

问题是SetElementValue正在添加一个“file”元素,然后在循环遍历foreach循环时覆盖该值,因此只有列表框中的最后一个值存储在XML文件中。添加新元素解决了问题。下面的代码是有用的:

            // Remove all "file" elements from selected record
            uElem.Descendants("file").ToList().ForEach(i => i.Remove());
            // Now add the elements from the listbox
            foreach (string s in lbFilesToProcess.Items)
            {
                // Add the new "file" element with the values from the listbox
                uElem.Element("filestoProcess").Add(new XElement("file", s));
            }