Выберите значение в Xelement в Xelement? - PullRequest
0 голосов
/ 21 августа 2011

Для приложения Webradio в Windows Phone я пытаюсь прочитать XML-файл с данными, но у меня проблема с конкретным полем. Файл XML выглядит следующим образом:

<brands xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
    <brandgroup>
        <brand>
            <code>blabla</code>
            <name>blabla</name>
            <logo>blabla</logo>
            <websiteurl>blabla</websiteurl>
            <audiostreams>
                <audiostream streamurl="www.1.mp3" codec="mp3" streamrate="low"/>
                <audiostream streamurl="www.2.mp3" codec="mp3" streamrate="med" default="true"/>
                <audiostream streamurl="www.3.mp3" codec="mp3" streamrate="high"/>
            </audiostreams>
        </brand>
        <brand>
        </brand>
    </brandgroup>
    other 'brandgroups' with other 'brand'
</brand>

С помощью следующего кода я могу получить имя, код и веб-сайт и объект Class Station для каждого бренда в каждой группе брендов.


XDocument loadedData = XDocument.Load("netten.xml");
var data = from query in loadedData.Descendants("brand") 
           select new Station
           {
               Name = (string)query.Element("name"),
               Code = (int)query.Element("code"),
               Website = (string)query.Element("websiteurl"), 
           };

Однако я не могу найти способ получить аудиопоток. Элемент audiostreams имеет 3 дочерних элемента audiostream, где мне нужен streamurl.

Лучше всего было бы хранить 3 потока, чтобы потом можно было изменить качество. Тогда мне нужно иметь поле в Class Station:

String[] streamurls = {www.1.mp3, www.2.mp3, www.3.mp3};

и сохраните там 3 потока, чтобы выбрать их позже. Я пробовал кое-что опубликованное здесь, связанное с XML, Attribute и XElement, но не могу заставить его работать.

Есть ли кто-нибудь, кто знает способ?

Кстати, я не совсем понимаю, как здесь выделить код и прочее, надеюсь, это работает, иначе мне очень жаль ...

Ответы [ 2 ]

2 голосов
/ 21 августа 2011

Если вам действительно нужны только URL-адреса, вы можете сделать что-то вроде (при условии, что свойство типа IEnumerable<string> (или string[]) называется StreamUrls на Station):

from brand in loadedData.Descendants("brand") 
select new Station
{
    Name = (string)brand.Element("name"),
    Code = (int)brand.Element("code"),
    Website = (string)brand.Element("websiteurl"), 
    StreamUrls = brand.Element("audiostreams")
              .Elements("audiostream")
              .Attributes("streamurl")
              .Select(a => a.Value)
              .ToArray()
}

Если вы хотите получить другую информацию из элементов audiostream, объявите класс как:

class Stream
{
    public string Url { get; set; }
    public string Codec { get; set; }
    public string Rate { get; set; }
    public bool IsDefault { get; set; }
}

И тогда запрос будет выглядеть так:

from brand in loadedData.Descendants("brand") 
select new Station
{
    Name = (string)brand.Element("name"),
    Code = (int)brand.Element("code"),
    Website = (string)brand.Element("websiteurl"), 
    Streams =
        (from stream in brand.Element("audiostreams").Elements("audiostream")
        select new Stream
        {
            Url = (string)stream.Attribute("streamurl"),
            Codec = (string)stream.Attribute("codec"),
            Rate = (string)stream.Attribute("streamrate"),
            IsDefault = (string)stream.Attribute("default") == "true"
        }).ToArray()
}

(Если Codec и Rate могут иметь только определенные значения, выражение их как enum будет лучше, чем string.)

0 голосов
/ 21 августа 2011

Надеюсь, это полезно

    static void Main(string[] args)
    {
        XDocument loadedData = XDocument.Load("netten.xml");
        var data = from query in loadedData.Descendants("brand")
                   group query by new { A = query.Element("name"), B = query.Element("code"), C = query.Element("websiteurl"), D = query.Element("audiostreams") } into g
                   select new
                   {
                       Name = g.Key.A + "",
                       Code = g.Key.B + "",
                       Website = g.Key.C + "",

                       AudioStreams = g.Key.D.Elements("audiostream")
                                                .Attributes("streamurl")
                                                .Select(x => x.Value)
                                                .ToArray()

                   };


        foreach (var x in data)
        {
            Console.WriteLine(x.Name);
            Console.WriteLine(x.Code);
            Console.WriteLine(x.Website);

            foreach (var url in x.AudioStreams)
                Console.WriteLine(url);
        }


        Console.ReadKey();
    }
}

XML-файл:

<?xml version="1.0" encoding="utf-8" ?>
<brands xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
  <brandgroup>
    <brand>
      <code>blabla</code>
      <name>blabla</name>
      <logo>blabla</logo>
      <websiteurl>blabla</websiteurl>
      <audiostreams>
        <audiostream streamurl="www.1.mp3" codec="mp3" streamrate="low"/>
        <audiostream streamurl="www.2.mp3" codec="mp3" streamrate="med" default="true"/>
        <audiostream streamurl="www.3.mp3" codec="mp3" streamrate="high"/>
      </audiostreams>
    </brand>
  </brandgroup>
  <brandgroup>
    <brand>
      <code>blabla2</code>
      <name>blabla2</name>
      <logo>blabla</logo>
      <websiteurl>blabla2</websiteurl>
      <audiostreams>
        <audiostream streamurl="www.4.mp3" codec="mp3" streamrate="low"/>
        <audiostream streamurl="www.5.mp3" codec="mp3" streamrate="med" default="true"/>
        <audiostream streamurl="www.6.mp3" codec="mp3" streamrate="high"/>
      </audiostreams>
    </brand>
  </brandgroup>
</brands>
...