2

以下のxmlを解析しようとしています:

<Item status="SUCCESS" message="">
   <ItemDate>12/21/2012
      <ItemType>MyType1
         <ItemUrl title="ItemTitle">http://www.itemurl1.com</ItemUrl>
      </ItemType>
   </ItemDate>
   <ItemDate>12/22/2012
      <ItemType>MyType2
         <ItemUrl title="Item2Title">http://www.itemurl2.com</ItemUrl>
      </ItemType>
   </ItemDate>
</Item>

ご覧のとおり、この xml を呼び出すことができるかどうかはわかりませんが、これはレガシー サービスから取得したものです。私が求めているのは、これを解析してオブジェクト グラフにロードすることです。私のオブジェクトモデルは以下の通りです:

 public class Item
    {
        public string Date { get; set; }
        public string Type { get; set; }
        public string Url { get; set; }
        public string Title { get; set; }
    }

したがって、基本的に、上記の xml/string の解析が完了すると、Item オブジェクトのコレクションが取得されます。コードスニペットでこれを達成する方法を教えてください。

XDocument でやってみましたが、xml の構造が独特でできませんでした。

ありがとう -マイク

4

3 に答える 3

2
XDocument xdoc = XDocument.Load(path_to_xml);
var query = from date in xdoc.Descendants("ItemDate")
            let type = date.Element("ItemType")
            let url = type.Element("ItemUrl")
            select new Item()
            {
                ItemDate = ((XText)date.FirstNode).Value,
                ItemType = ((XText)type.FirstNode).Value,
                ItemUrl = (string)url,
                ItemTitle = (string)url.Attribute("title"),
            };
于 2012-12-14T11:54:05.383 に答える
1

送信されたxmlにノードItemが1回しかないため、lazyberezovskyのコードから取得するItemは1つだけです。そしてそれは正しいです。アイテムを取得したいのですが、ItemDateノードでロードしたいと思います。これを行うには、次の変更されたコードを使用します。

XDocument xdoc = XDocument.Load(new StringReader(xml));
var query = from i in xdoc.Descendants( "ItemDate" )
                    let date = i
                    let type = date.Element("ItemType")
                    let url = type.Element("ItemUrl")
                    select new Item()
                            {
                                Date = ((XText) date.FirstNode).Value,
                                Type = ((XText) type.FirstNode).Value,
                                Url = (string) url,
                                Title = (string) url.Attribute("title"),
                            };
        var items = query.ToList();
于 2012-12-14T12:47:27.103 に答える
1

lazyberezovsky's Linq2Xmlプロジェクションの代わりに、Xmlをロードする前にXml Transformを使用してフラット化を行うことも検討できます。

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
                version="1.0"
                >
    <xsl:output omit-xml-declaration="yes" method="xml" version="1.0" indent="yes" />

    <xsl:template match="/">
        <Items>
            <xsl:apply-templates select="Item/ItemDate" />
        </Items>
    </xsl:template>

    <xsl:template match="ItemDate">
        <Item>
            <xsl:attribute name="ItemDate">
                <xsl:value-of select="normalize-space(./text()[1])" />
            </xsl:attribute>
            <xsl:attribute name="ItemType">
                <xsl:value-of select="normalize-space(ItemType/text()[1])" />
            </xsl:attribute>
            <xsl:attribute name="ItemUrl">
                <xsl:value-of select="normalize-space(ItemType/ItemUrl/text()[1])" />
            </xsl:attribute>
            <xsl:attribute name="ItemTitle">
                <xsl:value-of select="normalize-space(ItemType/ItemUrl/@title)" />
            </xsl:attribute>
        </Item>
    </xsl:template>
</xsl:stylesheet>

これにより、次の Xml が生成されます。これは、XmlDocumentの[XmlAttribute] 属性を使用するなど、簡単に逆シリアル化できます。

<Items>
  <Item ItemDate="12/21/2012" ItemType="MyType1" ItemUrl="http://www.itemurl1.com" ItemTitle="ItemTitle" />
  <Item ItemDate="12/22/2012" ItemType="MyType2" ItemUrl="http://www.itemurl2.com" ItemTitle="Item2Title" />
</Items>
于 2012-12-14T12:12:53.723 に答える