C# 将Linq转换为xml检索xml部分

C# 将Linq转换为xml检索xml部分,c#,xml,linq,linq-to-xml,C#,Xml,Linq,Linq To Xml,下面这段XML是更大的XML的一部分。我一次又一次地尝试获取地址行1、2、3以及城市和州、邮政编码和国家。我想把它功能化,这样我就可以选择 这些地址基于InvoiceHeader id=“XXXX”但是我一直在撞墙。 我尝试了下面的查询或类似的查询,但我一直得到一个错误对象引用,没有设置为对象的实例 这是我的问题,有人能指出我明显的错误吗 IEnumerable<string> partNos = from item in PurchaseOrderXml.

下面这段XML是更大的XML的一部分。我一次又一次地尝试获取地址行1、2、3以及城市和州、邮政编码和国家。我想把它功能化,这样我就可以选择 这些地址基于InvoiceHeader id=“XXXX”但是我一直在撞墙。 我尝试了下面的查询或类似的查询,但我一直得到一个错误对象引用,没有设置为对象的实例

这是我的问题,有人能指出我明显的错误吗

 IEnumerable<string> partNos =
            from item in PurchaseOrderXml.Descendants("RemitTo").Descendants("Address")
            where (string)item.Attribute("id").Value == "23951768"
            select (string)item;



<Invoice>
    <InvoiceHeader id="23951768" status="InProcess">
        <InvoiceName />
        <InvoiceNumber>23951768</InvoiceNumber>
        <InvoiceDate>2014-09-26 00:00:00.0</InvoiceDate>
        <DueDate>2014-10-26 00:00:00.0</DueDate>
        <SupplierInvoiceNo>534254504</SupplierInvoiceNo>
        <InvoiceType>Invoice</InvoiceType>
      <Supplier id="3825405">
        <ContactInfo type="main">
              <Address>
                <AddressLine lineNumber="1">Post </AddressLine>
                <AddressLine lineNumber="2">30 Street</AddressLine>
                <AddressLine lineNumber="3">30 Street</AddressLine>
                <City>Saint Louis</City>
                <State>MO</State>
                <PostalCode>63103-2530</PostalCode>
                <Country isoCountryCode="US">United States</Country>
            </Address>
        </ContactInfo>
    </Supplier>
        <BillTo>
        <Address>
            <AddressLine lineNumber="1">vvvv</AddressLine>
            <AddressLine lineNumber="2">vvvv</AddressLine>
            <City>Philadelphia</City>
            <State>PA</State>
            <PostalCode>19222</PostalCode>
            <Country isoCountryCode="US">United States</Country>
        </Address>
          </BillTo>
        <RemitTo>
            <Address>
                <AddressLine lineNumber="1">P O BOX 535182</AddressLine>
                <AddressLine lineNumber="2" />
                <AddressLine lineNumber="3" />
                <City>ATLANTA</City>
                <State>GA</State>
                <PostalCode>303535182</PostalCode>
                <Country isoCountryCode="US">United States</Country>
            </Address>
        </RemitTo>
     </InvoiceHeader>
</Invoice>
IEnumerable零件号=
从PurchaseOrderXml.substands(“汇款至”).substands(“地址”)中的项目
其中(字符串)item.Attribute(“id”).Value==“23951768”
选择(字符串)项;
23951768
2014-09-26 00:00:00.0
2014-10-26 00:00:00.0
534254504
发票联
邮递
30街
30街
圣路易斯
卫生官员
63103-2530
美国
VVV
VVV
费城
帕
19222
美国
邮政信箱535182
亚特兰大
GA
303535182
美国
试试这个:-

var result = xdoc.Root.Descendants("InvoiceHeader")
    .Where(x => x.Attribute("id").Value == "23951768")
    .SelectMany(x => x.Descendants("Address"))
    .Select(x =>
    {
       {
         var addressLine1 = x.Elements("AddressLine")
                        .FirstOrDefault(z => z.Attribute("lineNumber").Value == "1");
         var addressLine2 = x.Elements("AddressLine")
                        .FirstOrDefault(z => z.Attribute("lineNumber").Value == "2");
         var addressLine3 = x.Elements("AddressLine")
                        .FirstOrDefault(z => z.Attribute("lineNumber").Value == "3");
         return new
         {
            AddressLine1 = addressLine1 != null ? addressLine1.Value : String.Empty,
            AddressLine2 = addressLine2 != null ? addressLine1.Value : String.Empty,
            AddressLine3 = addressLine3 != null ? addressLine1.Value : String.Empty,
            City = x.Element("City").Value,
            State = x.Element("State").Value
            PostalCode = x.Element("PostalCode").Value,
            Country= x.Element("Country").Value,
          };
        }
    });

item range变量对应于没有id属性的Address元素。相反,您需要一个查询,该查询首先查找(或筛选)适当的InvoiceHeader,然后查找匹配的InvoiceHeader的Address元素desendents

以下是查找InvoiceHeader的示例:

var Header = PurchaseOrderXml.Descendants("InvoiceHeader")
.FirstOrDefault(header => (string)header.Attribute("id").Value == headerId);
您可以检查是否找到标头(
Header!=null
)。一旦有了标题,就可以在给定元素的范围内执行任何需要的操作。例如:

var RemitToAddress = Header.Descendants("RemitTo").Descendants("Address").FirstOrDefault();
您可能希望检查标头中的其他元素,因此将查询拆分为类似这样的部分可以保持您的意图清晰

还要注意,我使用了
子体
,但是如果它更符合您的模式,您也可以使用
元素

另一个例子是,要获取
AddressLine
元素并将它们连接起来,您可以尝试以下方法:

IEnumerable<string> AddressLines = RemitToAddress.Elements("AddressLine")
.OrderBy(line => (int)line.Attribute("lineNumber"))
.Select(line => line.Value);

var AddressText = string.Join("\n", AddressLines);
IEnumerable AddressLines=汇至Address.Elements(“AddressLine”)
.OrderBy(line=>(int)line.Attribute(“lineNumber”))
.选择(line=>line.Value);
var AddressText=string.Join(“\n”,地址行);

这似乎适用于我上面发布的较小的xml,但我对原始xml有问题。我得到的错误是序列不包含匹配元素?有什么想法吗?大型XML中的所有其他元素都与此类似吗?我考虑过您可以在
汇款到
节点中添加多个
地址
。序列不包含任何元素没有元素与您通过的条件匹配,您可以调试并告知我。我在更大的xml中有多个地址。我用一个更相关的示例更新了我的帖子。我认为多个地址可能是导致linq查询停止的原因?如果您现在查看我的xml,您会注意到我需要获取汇款地址的子代。@miguel-之所以这样,是因为您在第一个地址节点中没有地址行3,我们正在尝试获取它的值,这样它会在内部抛出null引用异常,在外部没有结果,只需检查null,然后获取值。我将更新该代码,所以我需要两个查询?我将更新以提供一个示例,但从概念上讲,您需要两个查询,把它们分开可能是有道理的。举个例子会很好。我在学钓鱼。迈克尔,这很干净,谢谢。还有一个问题。var REMITEOADRESS值以连接的3地址行结束。有没有办法在逻辑上把它们分开?对不起,我提出了一个愚蠢的问题。当然,您可以进一步查询
汇款地址
元素。我将用另一个例子更新。