使用LINQ进行XML解析时,有2个标签具有相同的名称但级别不同,并且不总是具有值

时间:2013-07-25 21:50:47

标签: c# xml linq xml-parsing

我有以下XML:

   <request>
       <book>
          <id>1833801</id>
          <title>The Yiddish Policemen's Union   </title> 
           <work>
               <id>1234</id>
               <name/>
           </work>
           <similar_books>
              <book><id>243859</id><title>Stations of Tide</title>                           <isbn>0380817616</isbn>
                 <authors><author><id>14454</id><name>Michael Swanwick</name></author>            </authors>
              </book>
          </similar_books>
          <authors>
              <author>
         <id>2715</id><name>Michael Chabon</name>
           <ratings_count>215884</ratings_count></author>
          </authors>
          <popular_shelves>
         <shelf name="jewish" count="104"/><shelf name="sci-fi" count="100"/>                   
         </popular_shelves>
       </book>                
     </request>

我希望所有标签都包含各自的值,我使用以下代码:

           HttpWebRequest oReq = (HttpWebRequest)WebRequest.Create(uriRoot);

                    HttpWebResponse resp = (HttpWebResponse)oReq.GetResponse();
                    log.Info(" (ISBN= " + isbn10 + ") Http request has response.");

                    if (resp.ContentType.StartsWith("application/xml", StringComparison.InvariantCultureIgnoreCase))
                    {
                        Stream resultStreamISBN = resp.GetResponseStream();
                        Encoding encode = System.Text.Encoding.GetEncoding("utf-8"); //encoding for non-latin chars
                        StreamReader responseReader = new StreamReader(resultStreamISBN, encode);

        XDocument xdoc = XDocument.Parse(responseReader.ReadToEnd());
        var books = (from u in xdoc.Descendants().Elements("book")
                     select new
                     {

                         id = (string)u.Element("title"),
                         title = (string)u.Element("title"),               
                         works = (from i in u.Elements("work")
                                  select new
                                  {
                                      work_best_book_id = (int)i.Element("id"),
                                      work_name = (string)i.Element("name"),

                                  }).ToList(),
                          authors = (from i in u.Elements("authors").Elements("author")
                                    select new
                                    {
                                        id = (int)i.Element("id"),
                                        name = (string)i.Element("name"),                                       
                                        rating = (int)i.Element("rating_count")
                                    }).ToList(),
                         popular_shelves = (from i in u.Elements("popular_shelves").Elements("shelf")
                                            select new
                                            {
                                                name = (string)i.Attribute("name"),
                                                count = (int)i.Attribute("count")
                                            }).ToList(),

                     }).ToList();

代码返回空值并且无法正常工作。我还应该注意,不同的xml文件可能没有所有标签的值。

有关如何改进代码的任何建议?

2 个答案:

答案 0 :(得分:0)

检查缺少的元素,如:

name = i.Element("name") == null ? null : i.Element("name")

对于值类型,您必须将它们更改为可空类型才能使其正常工作。

答案 1 :(得分:0)

你可以转换

id = (int)i.Element("id"),

id = (int?)i.Element("id"),

整个修改过的代码

// I am inserting the following line to demonstrate how I load the XML in test code
// Basically, I copied & pasted xml in a file OP gave named request.xml
//XDocument xdoc = XDocument.Load( @"d:\Data\request.xml");

修改

XDocument xdoc = XDocument.Load("http://www.goodreads.com/book/isbn?isbn=0007295685&key=lbScLXWyNGQ1q0BDoFFSg");

var books = (from u in xdoc.Descendants().Elements("book")
                select new
                {

                    id = (string)u.Element("title"),
                    title = (string)u.Element("title"),
                    works = (from i in u.Elements("work")
                            select new
                            {
                                work_best_book_id = (int?)i.Element("id"),
                                work_name = (string)i.Element("name"),

                            }).ToList(),
                    authors = (from i in u.Elements("authors").Elements("author")
                            select new
                            {
                                id = (int?)i.Element("id"),
                                name = (string)i.Element("name"),
                                rating = (int?)i.Element("rating_count")
                            }).ToList(),
                    popular_shelves = (from i in u.Elements("popular_shelves").Elements("shelf")
                                    select new
                                    {
                                        name = (string)i.Attribute("name"),
                                        count = (int?)i.Attribute("count")
                                    }).ToList(),

                }).ToList();