2012-07-27 64 views
0

我是LINQ to XML的新手,想知道是否有人可以帮助我构建以下查询。LINQ to XML不包含后代包含值的元素的示例

我想返回所有<response>元素,它们不包含含有“404”的后代<status>元素。

我的XML如下所示。在这种情况下,只应返回第一个<response>元素(和后代)。

<multistatus xmlns="DAV:"> 
    <response> 
    <href>/principals/users/test/</href> 
    <propstat> 
     <prop> 
     <calendar-home-set xmlns="urn:ietf:params:xml:ns:caldav"> 
      <href xmlns="DAV:">/calendars/__uids__/d817aaec-7d24-5b38-bc2f-6369da72cdd9</href> 
     </calendar-home-set> 
     </prop> 
     <status>HTTP/1.1 200 OK</status> 
    </propstat> 
    </response> 
    <response> 
    <href>/principals/users/test/calendar-proxy-write/</href> 
    <propstat> 
     <prop> 
     <calendar-home-set xmlns="urn:ietf:params:xml:ns:caldav" /> 
     </prop> 
     <status>HTTP/1.1 404 Not Found</status> 
    </propstat> 
    </response> 
    <response> 
    <href>/principals/users/test/calendar-proxy-read/</href> 
    <propstat> 
     <prop> 
     <calendar-home-set xmlns="urn:ietf:params:xml:ns:caldav" /> 
     </prop> 
     <status>HTTP/1.1 404 Not Found</status> 
    </propstat> 
    </response> 
</multistatus> 

回答

3

假设你的XML存储在字符串变量xml

XDocument document = XDocument.Parse(xml); 
XNamespace ns = document.Root.GetDefaultNamespace(); 

var responsesExcept404s = document 
    .Descendants(ns + "response") 
    .Where(x => !x.Descendants(ns + "status") 
        .Single() 
        .Value.Contains("404")); 

通知ns变量的用法 - 因为你的XML有默认命名空间集通过xmlns属性,有必要在使用LINQ to XML时指定该名称空间(例如在Descendants()方法中)。

然后,你可以简单地遍历结果,使这个超有用,它们输出到控制台:

responsesExcept404s.ToList().ForEach(Console.WriteLine); 
+0

嗯......你测试过了吗?对我来说,我得到空的结果。对于命名空间,我使用“DAV:”。 – 2012-07-27 21:36:58

+0

@JonathanWood是的,我做到了。我完全使用了你的问题中的XML。我的整个程序是一个简单的.NET 4.0 C#控制台应用程序,它包含将XML存储到字符串变量以及我的答案中的代码,仅此而已。你应该得到一个'回应'回来。 – 2012-07-27 21:41:28

+0

Doh!看起来我的名字空间错了。您的代码与广告一样工作! – 2012-07-27 22:28:14

0

试试这个:

Document doc = XDocument.Load(path); 
     XNamespace nsd = doc.Root.GetDefaultNamespace(); 
     var res = doc.Descendants(nsd +"response");     

     var filteredEle = new List<XElement>(); 

        foreach (var ele in res) 
        { 
         if (CheckEle(ele,nsd)) 
         { 
          filteredEle.Add(ele); 
         } 
        }  


    private bool CheckEle(XElement ele, XNamespace nsd) 
     { 
      return ele.Element(nsd + "propstat").Element(nsd + "status").Value != "HTTP/1.1 404 Not Found"; 
     } 
+0

谢谢,但正如我的问题所示,我的XML内容不是这样的。我比较的不是一个属性,它是一个子元素。 – 2012-07-27 21:11:16

+0

我已经编辑了答案,你现在可以试试。 – 2012-07-27 21:20:36

+0

谢谢,但这仍然不同。在这里,''是来自''的直接孩子。在我的XML中,事实并非如此。这就是为什么我将XML包含在问题中的原因。 – 2012-07-27 21:25:17

0

你可以使用LINQ语句如下图所示

XDocument doc = XDocument.Parse(xml); 
List<XElement> responseWithOut404 = 
    (from element in doc.Descendants("response") 
    let xElement = element.Descendants("status").First() 
    where !xElement.Value.Contains("404") 
    select element) 
    .ToList(); 
1

这里亚去:(使用XPath作弊)

 XDocument xdoc = XDocument.Load(new FileStream("XMLFile2.xml", FileMode.Open, FileAccess.Read)); 
     XPathNavigator nav = xdoc.CreateNavigator(); 
     var nsm = new XmlNamespaceManager(nav.NameTable);    
     nsm.AddNamespace("s", "DAV:"); 
     var nodes = xdoc.XPathSelectElements("s:multistatus/s:response[.//*[name(.)='status' and .='HTTP/1.1 404 Not Found']]", nsm); 
+0

谢谢。看起来“毛茸茸”。 :)不会直接在我的情况下工作,因为我有'XElement'而不是'XDocument'。 – 2012-07-27 22:30:22

2
XDocument xDoc = XDocument.Parse(xml); 
XNamespace ns = XNamespace.Get("DAV:"); 
var responses = xDoc.Descendants(ns + "status") 
        .Where(s => !s.Value.Contains(" 404 ")) 
        .Select(s => s.Parent.Parent);