2011-12-18 8 views
8

Biorąc pod uwagę ten XML w ciąg o nazwie "string":Jak mogę iterować przez węzły podrzędne o określonej nazwie w Nokogiri XML DocumentFragment?

<Guide> 
    <Master> 
    <Part>12345</Part> 
    <Sub> 
     <Name>A</Name> 
    </Sub> 
    <Sub> 
     <Name>B</Name> 
    </Sub> 
    </Master> 
    <Master> 
    <Part>XYZABC</Part> 
    <Sub> 
     <Name>A</Name> 
    </Sub> 
    <Sub> 
     <Name>C</Name> 
    </Sub> 
    </Master> 
</Guide> 

I ta linia kodu:

bgdoc = Nokogiri::XML::DocumentFragment.parse(xstring.to_xml) 

Chcę pętli wszystkich węzłach nazwanych "Część".

Próbowałem następujące:

bgdoc.xpath("//Part").each do |node| 

oraz:

bgdoc.children.each do |node| 
    next unless node.name=="Part" 

Ale to nie działa.

Odpowiedz

8

Problemem jest to, że podczas analizowania XML jako fragment zwraca częściowego dokument XML, czyli do DocumentFragment, które nie mają korzenia:

1.9.2-p290 :002 > doc = Nokogiri::XML::DocumentFragment.parse('<a><b>foo</b></a>').root 
NoMethodError: undefined method `root' for #<Nokogiri::XML::DocumentFragment:0x00000100b34448> 
    from (irb):2 
    from /Users/greg/.rvm/rubies/ruby-1.9.2-p290/bin/irb:16:in `<main>' 

natomiast dokument XML ma pełne korzeń:

1.9.2-p290 :003 > doc = Nokogiri::XML('<a><b>foo</b></a>').root 
=> #<Nokogiri::XML::Element:0x8058b350 name="a" children=[#<Nokogiri::XML::Element:0x80587b10 name="b" children=[#<Nokogiri::XML::Text:0x80587818 "foo">]>]> 

domyślnie Nokogiri będzie szukał od korzenia dokumentu z XPath jak //Path:

1.9.2-p290 :004 > doc = Nokogiri::XML('<a><Path>foo</Path></a>').search('//Path') 
=> [#<Nokogiri::XML::Element:0x8055465c name="Path" children=[#<Nokogiri::XML::Text:0x805543c8 "foo">]>] 

Ale to nie z fragmentem z powodu brakującego root:

1.9.2-p290 :005 > doc = Nokogiri::XML::DocumentFragment.parse('<a><Path>foo</Path></a>').search('//Path') 
=> [] 

Sztuką jest powiedzieć Nokogiri gdzie szukać gdy ma do czynienia z fragmentem. Korzystanie względną wyszukiwania lub symbol wieloznaczny:

1.9.2-p290 :006 > doc = Nokogiri::XML::DocumentFragment.parse('<a><Path>foo</Path></a>').search('.//Path') 
=> [#<Nokogiri::XML::Element:0x8053c69c name="Path" children=[#<Nokogiri::XML::Text:0x8053c46c "foo">]>] 

lub

1.9.2-p290 :007 > doc = Nokogiri::XML::DocumentFragment.parse('<a><Path>foo</Path></a>').search('*//Path') 
=> [#<Nokogiri::XML::Element:0x8052a208 name="Path" children=[#<Nokogiri::XML::Text:0x80529fec "foo">]>] 
1
bgdoc = Nokogiri::XML::DocumentFragment.parse(<<EOF) 
    <xml stuff> 
EOF 

bgdoc.xpath(".//Part").each do |node| 
    # some instruction 
end 
1

Jeśli masz tylko łańcuch, po prostu parsowania z Nokogiri:XML Zamiast:

bgdoc = Nokogiri::XML.parse(string) 

dadzą Ci pierwiastek z które możesz użyć xpath //Part zgodnie z oczekiwaniami.