Linq查询如果元素不存在则更新XML文件

时间:2012-04-21 07:23:37

标签: c# xml linq windows-phone-7

目前正在开发一款wp7应用程序,它非常基础。用户有一个计数器,如果XML文件中当天存在日期元素,则更新计数,如果没有为该日创建新的日期元素并将计数作为值。

我的priblem是,如果创建一个新的XML文件,一切正常,当前日期元素更新没问题,但如果我第二天测试,则会创建一个新元素,但是当我想更新计数时,添加了一个新的日期元素。我没有得到这个,因为所有代码都适用于新文件,但是如果文件是一天的话,它不是出于某种原因。

XML代码

<?xml version="1.0" encoding="utf-8"?>
<Countlog>
 <date Count="9">4/21/2012</date>
 <date Count="4">4/21/2012</date>
 <date Count="18">4/21/2012</date>
</Countlog>

C#

private void save_btn_Click(object sender, RoutedEventArgs e)
    {
        String _count = Count_tb.Text;
        String s_todaysdate = todaysdate.Date.ToShortDateString();

        IsolatedStorageFileStream isoStream = new IsolatedStorageFileStream("Countlog.xml", FileMode.Open, myIsolatedStorage);
        StreamReader reader = new StreamReader(isoStream);
        XDocument _xml = XDocument.Load(reader);
        isoStream.Close();

        var query = from r in _xml.Descendants("Countlog")
                    where r.Element("date").Value == (DateTime.Now.ToShortDateString())
                    select r.Element("date");


        if (!query.Any())
        {
            XElement Addnewdate = new XElement("date", s_todaysdate, new XAttribute("Count", _count));
            _xml.Root.Add(Addnewdate);
            MessageBox.Show("no matching date");
        }
        else
        {
            foreach (XElement _date in query)
            {
                _date.Attribute("Count").Value = _count.ToString();
                MessageBox.Show("Updating date");

            }
        }

        IsolatedStorageFileStream isoStreamsave = new IsolatedStorageFileStream("Countlog.xml", FileMode.Truncate, myIsolatedStorage);

        _xml.Save(isoStreamsave);
        isoStreamsave.Close();
    }





    private void Pivot_SelectionChanged(object sender, SelectionChangedEventArgs e)
    {
        if (pivotholder.SelectedIndex == 1)
        {
            IsolatedStorageFileStream isoFileStream2 = myIsolatedStorage.OpenFile("Countlog.xml", FileMode.Open);
            StreamReader reader = new StreamReader(isoFileStream2);
            XML_result.Text = reader.ReadToEnd();
            reader.Close();
        }

    }

如果您需要更多信息,请告诉我。这是我在过去几年潜伏之后第一次在这里发帖。

干杯

乔恩

2 个答案:

答案 0 :(得分:1)

好的,我已经解决了。这个查询:

var query = from r in _xml.Descendants("Countlog")
            where r.Element("date").Value == (DateTime.Now.ToShortDateString())
            select r.Element("date");

仅在第一个 date元素具有正确值时匹配。您正在迭代所有Countlog个元素(其中只有一个元素),并查找第一个date元素(因为这是Element(...)所做的)。

您可以将其更改为简单使用:

var query = _xml.Root.Elements("date")
                     .Where(x => x.Value == (DateTime.Now.ToShortDateString())

但是,我建议采用另一种格式:

var date = DateTime.Today;
var query = _xml.Root.Elements("date")
                .Where(x => (DateTime) x.Value == date);

然后添加一个新元素:

XElement element = new XElement("date",
                      new XAttribute("count", count),
                      date);

或者更新一个:

element.Attribute("count").SetValue(count);

这使用LINQ to XML的数据类型处理,而不是明确地将所有内容转换为字符串。

答案 1 :(得分:0)

1)您当前表单中的查询将搜索当时的实际日期

where r.Element("date").Value == (DateTime.Now.ToShortDateString())

另一方面,我相信,您要查询的内容是您存储在s_todaysdate变量中的其他日期。

2)正如Jon Skeet已经指出的那样,你假设只有一个<date>元素。    通过_xml.Root.Elements("date")开始查询。

因此,最终查询需要修改为:

var query = from date in _xml.Root.Elements("date")
            where date.Value == s_todaysdate
            select date;