使用xmlns属性(命名空间)查询XDocument

时间:2013-04-15 14:57:50

标签: c# xml linq-to-xml

我尝试查询visual studio * .csproj文件中的元素。我创建了一个简短的例子来说明问题:

    // Working
    string xml1 = @"<Project ToolsVersion='4.0'>
                      <ItemGroup Label='Usings'>
                        <Reference Include='System' />
                        <Reference Include='System.Xml' />
                      </ItemGroup>
                    </Project>"; 
    // Not working
    string xml2 = @"<Project ToolsVersion='4.0' xmlns='http://schemas.microsoft.com/developer/msbuild/2003'>
                      <ItemGroup Label='Usings'>
                        <Reference Include='System' />
                        <Reference Include='System.Xml' />
                      </ItemGroup>
                    </Project>";

    XDocument doc = XDocument.Parse(xml2);

    foreach (XElement element in doc.Descendants("ItemGroup"))
    {
        Console.WriteLine(element);
    }

字符串xml1工作正常,xml2不返回任何内容。这些字符串之间的唯一区别是文档根目录中的xmlns属性。

如何查询包含xmlns属性的文档? 当xml文档包含xmlns属性时,为什么会出现问题?

2 个答案:

答案 0 :(得分:19)

  

当xml文档包含xmlns属性时,为什么会出现问题?

如果你明白这意味着什么,那就不是了:)基本上你已经为所有元素应用了一个默认的名称空间URI“http://schemas.microsoft.com/developer/msbuild/2003”。因此,在查询时,您还需要指定该命名空间。幸运的是,LINQ to XML非常简单:

XNamespace ns = "http://schemas.microsoft.com/developer/msbuild/2003";
XDocument doc = XDocument.Parse(xml2);
foreach (XElement element in doc.Descendants(ns + "ItemGroup"))
{
    Console.WriteLine(element);
}

答案 1 :(得分:4)

没有必要事先知道命名空间。 您可以编写适用于Xmls的代码,因为您可以从XElement获取默认命名空间。

XDocument doc = XDocument.Parse(xml2);
XNamespace ns = doc.Root.GetDefaultNamespace();
foreach (XElement element in doc.Descendants(ns + "ItemGroup"))
{
    Console.WriteLine(element);
}

我还编写了一个扩展方法来解析任何XObject(XElement,XDocument等)中的XName。

使用扩展方法而不是GetDefaultNamespace的好处是您不必检查是否已经提供了另一个命名空间。

public static XName ResolveName(this XObject xObj, XName name)
{
    //If no namespace has been added, use default namespace anyway
    if (string.IsNullOrEmpty(name.NamespaceName))
    {
        name = xObj.Document.Root.GetDefaultNamespace() + name.LocalName;
    }
    return name;
}

您可以像这样使用

XDocument doc = XDocument.Parse(xml2);
foreach (XElement element in doc.Descendants(doc.ResolveName("ItemGroup")))
{
    Console.WriteLine(element);
}

我认为LINQ to XML是一个很棒的API。但我认为很明显,如果我不提供命名空间,那么我总是指默认命名空间。我没有看到任何理由为什么LINQ to XML不会以这种方式运行。这有点让我恼火的缺点。第一次作为LINQ to XML的初学者,当我忘记提供默认命名空间时,我几个小时都不知道自己做错了什么。