2013-03-05 226 views
1

我有下面的XML:的LINQ to XML子查詢

<Event ID="1"..... > 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
</Event> 
<Event ID="2"..... > 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
</Event> 
<Event ID="3"..... > 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
    <SubEvent update="DATETIME" /> 
</Event> 

我想刪除「事件」元素,其中的子事件全部更新屬性比提供DATETIME低。

如果例如一個DATETIME較高,則不應刪除該元素。

+0

爲什麼你覺得呢?你覺得它太慢了嗎? – 2013-03-05 13:47:48

+2

你認爲LINQ to XML如何工作?有魔力? :)它也遍歷元素。 – 2013-03-05 13:50:19

+0

我意識到我的最後一條評論可能沒有幫助你,所以這裏有一些額外的建議:儘量減少循環。例如,如果「Event」的第一個「SubEvent」大於提供的日期,則可以停止處理該「Event」。沒有必要遍歷剩餘的SubEvent元素。但除此之外,你可以做的不多。 – 2013-03-05 13:57:54

回答

3

簡單地選擇所有這些事件,並從文件中刪除:

DateTime date = DateTime.Now; 
XDocument xdoc = XDocument.Load(path_to_xml); 
xdoc.Descendants("Event") 
    .Where(e => e.Elements().All(se => (DateTime)se.Attribute("update") < date)) 
    .Remove(); 
xdoc.Save(path_to_xml); 

正常工作與下面的XML:

<?xml version="1.0" encoding="utf-8" ?> 
<Events> 
    <Event ID="1" > 
    <SubEvent update="2013-02-05T17:06:23.8962976+03:00" /> 
    <SubEvent update="2013-03-08T17:06:23.8962976+03:00" /> 
    </Event> 
    <Event ID="2"> 
    <SubEvent update="2013-01-05T17:06:23.8962976+03:00" /> 
    <SubEvent update="2013-01-05T17:06:23.8962976+03:00" /> 
    <SubEvent update="2013-02-05T17:06:23.8962976+03:00" /> 
    </Event> 
    <Event ID="3"> 
    <SubEvent update="2013-03-05T17:06:23.8962976+03:00" /> 
    <SubEvent update="2013-04-05T17:06:23.8962976+03:00" /> 
    </Event> 
</Events> 
+0

我很好奇爲什麼在你的示例''刪除'工作正常,但是當我在'我的foreach'像'el.Remove()'它只適用於第一個元素? – aush 2013-03-05 14:10:28

+0

@aush堅持,我會檢查 – 2013-03-05 14:12:33

+1

@aush看起來像枚舉中斷時修改文檔。如果你將在foreach語句中執行'query.ToList()',那麼所有的都會正常工作。如果你不會刪除元素(只是枚舉,那麼一切都很好)。你可能想知道爲什麼它可以與'Enumerable '擴展一起使用? :)因爲內部還創建了新的元素列表:'foreach(T本地新列表(源))' – 2013-03-05 14:27:12

0

在想你是不會錯過你的XML

var xDoc = // your XDocument 
var toDel = new List<XElement>(); 
foreach(var el in xDoc.Root.Elements("Event").Where(e => e.Elements("SubEvent").All(xel => xel.Attribute("update").Value == "DATETIME"))) 
{ 
    toDel.Add(el); 
} 
toDel.ForEach(e => e.Remove()); 

的根必須更換的條件內.All()到理想的。

0

試試這個:

 var doc = XDocument.Load(@"D:\input.xml"); 
     var x = new List<XElement>(); 
     foreach (var xElement in doc.Root.Elements("Event")) 
     { 
      DateTime maxDt = DateTime.MinValue; 

      foreach (var element in xElement.Elements("SubEvent")) 
      { 
       var attributeValue = element.Attributes("update").FirstOrDefault(); 

       if (attributeValue == null) continue; 

       DateTime dt; 
       if (!DateTime.TryParse(attributeValue.Value, out dt)) continue; 

       if (maxDt < dt) maxDt = dt; 
      } 

      xElement.RemoveNodes(); 
      xElement.Add(new XElement("SubEvent", new XAttribute("update", maxDt.ToString("yyyy-MM-dd HH:mm:ss")))); 
      x.Add(new XElement(xElement)); 

     } 

     new XDocument(new XElement("ev", x)).Save(@"D:\output.xml");