Разбор простого XML с помощью Nokogiri - PullRequest
17 голосов
/ 15 октября 2010

У меня есть следующий XML:

<links>

  <item>
    <title>Title 1</title>
    <url>http://www.example.com/url-1</url>
  </item>

  <item>
   <title>Title 2</title>
   <url>http://www.example.com/url-2</url>
  </item>

  <item>
    <title>Title 3</title>
    <url>http://www.example.com/url-3</url>
  </item>

</links>

И я хотел бы преобразовать его в список HTML:

<ul>
  <li><a href="http://www.example.com/url-1">Title 1</a></li>
  <li><a href="http://www.example.com/url-2">Title 2</a></li>
  <li><a href="http://www.example.com/url-3">Title 3</a></li>
</ul>

В настоящее время у меня есть это:

Контроллер:

require 'nokogiri'
doc = Nokogiri::XML(...)

@links = doc.xpath('//links/item').map do |i|
  {'title' => i.xpath('//title'), 'url' => i.xpath('//url')}
end

Шаблон:

<ul>
  <% @links.each do |l| %>
    <li><a href="<%= l['url'] %>"><%= l['title'] %></a></li>
  <% end %>
</ul> 

Результирующий HTML:

<ul>
  <li><a href="http://www.example.com/url-1http://www.example.com/url-2http://www.example.com/url-3">Title 1Title 2Title 3</a></li>
  <li><a href="http://www.example.com/url-1http://www.example.com/url-2http://www.example.com/url-3">Title 1Title 2Title 3</a></li>
  <li><a href="http://www.example.com/url-1http://www.example.com/url-2http://www.example.com/url-3">Title 1Title 2Title 3</a></li>
</ul>

Что я делаю не так? Есть ли более оптимальный способ сделать это?

Ответы [ 2 ]

27 голосов
/ 15 октября 2010

Заменить :

@links = doc.xpath('//links/item').map do |i| 
  {'title' => i.xpath('//title'), 'url' => i.xpath('//url')} 

на :

@links = doc.xpath('//links/item').map do |i| 
  {'title' => i.xpath('title'), 'url' => i.xpath('url')} 

Объяснение :

//title 

и

//url

являются абсолютными выражениями XPath, и они выбирают все (соответственно) title и все url элементы в документе XML.

Сравните это с :

title

и

url

Это относительно выражений XPath и выберите все (соответственно) title и url только дочерние узлы current .

6 голосов
/ 15 октября 2010

Проблема в том, что Xpath //title ищет заголовки из корня документа и возвращает все теги title.Использование Xpath title выполняет поиск в контексте данного узла, как вы хотите.То же самое на url.

@links = doc.xpath('//links/item').map do |i|
  {'title' => i.xpath('title'), 'url' => i.xpath('url')}
end
...